web-agent-bridge 1.1.2 → 2.0.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.
Files changed (94) hide show
  1. package/LICENSE +21 -21
  2. package/README.ar.md +446 -446
  3. package/README.md +780 -844
  4. package/bin/cli.js +80 -80
  5. package/bin/wab.js +80 -80
  6. package/examples/bidi-agent.js +119 -119
  7. package/examples/mcp-agent.js +94 -94
  8. package/examples/next-app-router/README.md +44 -0
  9. package/examples/puppeteer-agent.js +108 -108
  10. package/examples/saas-dashboard/README.md +55 -0
  11. package/examples/shopify-hydrogen/README.md +74 -0
  12. package/examples/vision-agent.js +171 -171
  13. package/examples/wordpress-elementor/README.md +77 -0
  14. package/package.json +69 -78
  15. package/public/.well-known/ai-assets.json +59 -0
  16. package/public/admin/login.html +84 -84
  17. package/public/ai.html +196 -0
  18. package/public/cookies.html +208 -208
  19. package/public/css/premium.css +317 -0
  20. package/public/css/styles.css +1235 -1235
  21. package/public/dashboard.html +704 -704
  22. package/public/demo.html +259 -0
  23. package/public/docs.html +585 -585
  24. package/public/feed.xml +89 -0
  25. package/public/index.html +495 -332
  26. package/public/js/auth-nav.js +31 -31
  27. package/public/js/auth-redirect.js +12 -12
  28. package/public/js/cookie-consent.js +56 -56
  29. package/public/js/wab-demo-page.js +721 -0
  30. package/public/js/ws-client.js +74 -74
  31. package/public/llms-full.txt +309 -0
  32. package/public/llms.txt +85 -0
  33. package/public/login.html +83 -83
  34. package/public/openapi.json +580 -0
  35. package/public/premium-dashboard.html +2487 -0
  36. package/public/premium.html +791 -0
  37. package/public/privacy.html +295 -295
  38. package/public/register.html +103 -103
  39. package/public/robots.txt +87 -0
  40. package/public/script/wab-consent.d.ts +36 -0
  41. package/public/script/wab-consent.js +104 -0
  42. package/public/script/wab-schema.js +131 -0
  43. package/public/script/wab.d.ts +108 -0
  44. package/public/script/wab.min.js +234 -0
  45. package/public/sitemap.xml +93 -0
  46. package/public/terms.html +254 -254
  47. package/public/video/tutorial.mp4 +0 -0
  48. package/script/ai-agent-bridge.js +1558 -1513
  49. package/sdk/README.md +55 -55
  50. package/sdk/index.d.ts +118 -0
  51. package/sdk/index.js +257 -203
  52. package/sdk/package.json +14 -14
  53. package/sdk/schema-discovery.js +83 -0
  54. package/server/config/secrets.js +94 -92
  55. package/server/index.js +0 -9
  56. package/server/middleware/adminAuth.js +30 -30
  57. package/server/middleware/auth.js +41 -41
  58. package/server/middleware/rateLimits.js +24 -24
  59. package/server/migrations/001_add_analytics_indexes.sql +7 -7
  60. package/server/migrations/002_premium_features.sql +418 -0
  61. package/server/models/adapters/index.js +33 -33
  62. package/server/models/adapters/mysql.js +183 -183
  63. package/server/models/adapters/postgresql.js +172 -172
  64. package/server/models/adapters/sqlite.js +7 -7
  65. package/server/models/db.js +561 -561
  66. package/server/routes/admin-premium.js +671 -0
  67. package/server/routes/admin.js +247 -247
  68. package/server/routes/api.js +131 -138
  69. package/server/routes/auth.js +51 -51
  70. package/server/routes/billing.js +45 -45
  71. package/server/routes/discovery.js +406 -329
  72. package/server/routes/license.js +240 -240
  73. package/server/routes/noscript.js +543 -543
  74. package/server/routes/premium-v2.js +686 -0
  75. package/server/routes/premium.js +724 -0
  76. package/server/routes/wab-api.js +476 -476
  77. package/server/services/agent-memory.js +625 -0
  78. package/server/services/email.js +204 -204
  79. package/server/services/fairness.js +420 -420
  80. package/server/services/plugins.js +747 -0
  81. package/server/services/premium.js +1883 -0
  82. package/server/services/self-healing.js +843 -0
  83. package/server/services/stripe.js +192 -192
  84. package/server/services/swarm.js +788 -0
  85. package/server/services/vision.js +871 -0
  86. package/server/utils/cache.js +125 -125
  87. package/server/utils/migrate.js +81 -81
  88. package/server/utils/secureFields.js +50 -50
  89. package/server/ws.js +101 -101
  90. package/docs/DEPLOY.md +0 -118
  91. package/docs/SPEC.md +0 -1540
  92. package/wab-mcp-adapter/README.md +0 -136
  93. package/wab-mcp-adapter/index.js +0 -555
  94. package/wab-mcp-adapter/package.json +0 -17
@@ -1,172 +1,172 @@
1
- /**
2
- * PostgreSQL Adapter for WAB
3
- *
4
- * Prerequisites: npm install pg
5
- * Set DATABASE_URL=postgres://user:pass@host:5432/wab
6
- *
7
- * This adapter implements the same interface as the SQLite adapter
8
- * so it can be used as a drop-in replacement.
9
- */
10
-
11
- const { Pool } = require('pg');
12
- const bcrypt = require('bcryptjs');
13
- const { v4: uuidv4 } = require('uuid');
14
-
15
- const pool = new Pool({ connectionString: process.env.DATABASE_URL });
16
-
17
- // Initialize tables
18
- async function initDB() {
19
- await pool.query(`
20
- CREATE TABLE IF NOT EXISTS users (
21
- id TEXT PRIMARY KEY,
22
- email TEXT UNIQUE NOT NULL,
23
- password TEXT NOT NULL,
24
- name TEXT NOT NULL,
25
- company TEXT,
26
- created_at TIMESTAMPTZ DEFAULT NOW(),
27
- updated_at TIMESTAMPTZ DEFAULT NOW()
28
- );
29
-
30
- CREATE TABLE IF NOT EXISTS sites (
31
- id TEXT PRIMARY KEY,
32
- user_id TEXT NOT NULL REFERENCES users(id) ON DELETE CASCADE,
33
- domain TEXT NOT NULL,
34
- name TEXT NOT NULL,
35
- description TEXT,
36
- tier TEXT DEFAULT 'free' CHECK(tier IN ('free','starter','pro','enterprise')),
37
- license_key TEXT UNIQUE NOT NULL,
38
- api_key TEXT UNIQUE,
39
- config JSONB DEFAULT '{}',
40
- active BOOLEAN DEFAULT TRUE,
41
- created_at TIMESTAMPTZ DEFAULT NOW(),
42
- updated_at TIMESTAMPTZ DEFAULT NOW()
43
- );
44
-
45
- CREATE TABLE IF NOT EXISTS analytics (
46
- id SERIAL PRIMARY KEY,
47
- site_id TEXT NOT NULL REFERENCES sites(id) ON DELETE CASCADE,
48
- action_name TEXT NOT NULL,
49
- agent_id TEXT,
50
- trigger_type TEXT,
51
- success BOOLEAN,
52
- metadata JSONB DEFAULT '{}',
53
- created_at TIMESTAMPTZ DEFAULT NOW()
54
- );
55
-
56
- CREATE TABLE IF NOT EXISTS subscriptions (
57
- id TEXT PRIMARY KEY,
58
- user_id TEXT NOT NULL REFERENCES users(id) ON DELETE CASCADE,
59
- site_id TEXT NOT NULL REFERENCES sites(id) ON DELETE CASCADE,
60
- tier TEXT NOT NULL CHECK(tier IN ('free','starter','pro','enterprise')),
61
- status TEXT DEFAULT 'active' CHECK(status IN ('active','cancelled','expired','trial')),
62
- started_at TIMESTAMPTZ DEFAULT NOW(),
63
- expires_at TIMESTAMPTZ
64
- );
65
-
66
- CREATE INDEX IF NOT EXISTS idx_sites_domain ON sites(domain);
67
- CREATE INDEX IF NOT EXISTS idx_sites_license ON sites(license_key);
68
- CREATE INDEX IF NOT EXISTS idx_analytics_site ON analytics(site_id);
69
- CREATE INDEX IF NOT EXISTS idx_analytics_created ON analytics(created_at);
70
- `);
71
- }
72
-
73
- initDB().catch(console.error);
74
-
75
- function generateLicenseKey() {
76
- const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
77
- const segments = [];
78
- for (let s = 0; s < 4; s++) {
79
- let seg = '';
80
- for (let i = 0; i < 5; i++) seg += chars[Math.floor(Math.random() * chars.length)];
81
- segments.push(seg);
82
- }
83
- return `WAB-${segments.join('-')}`;
84
- }
85
-
86
- function generateApiKey() {
87
- return `wab_${uuidv4().replace(/-/g, '')}`;
88
- }
89
-
90
- // ─── User Operations ──────────────────────────────────────────────────
91
- async function registerUser({ email, password, name, company }) {
92
- const id = uuidv4();
93
- const hashed = bcrypt.hashSync(password, 12);
94
- await pool.query(
95
- 'INSERT INTO users (id, email, password, name, company) VALUES ($1, $2, $3, $4, $5)',
96
- [id, email, hashed, name, company || null]
97
- );
98
- return { id, email, name, company };
99
- }
100
-
101
- async function loginUser({ email, password }) {
102
- const { rows } = await pool.query('SELECT * FROM users WHERE email = $1', [email]);
103
- const user = rows[0];
104
- if (!user) return null;
105
- if (!bcrypt.compareSync(password, user.password)) return null;
106
- return { id: user.id, email: user.email, name: user.name, company: user.company };
107
- }
108
-
109
- // ─── Site Operations ──────────────────────────────────────────────────
110
- async function addSite({ userId, domain, name, description, tier }) {
111
- const id = uuidv4();
112
- const licenseKey = generateLicenseKey();
113
- const apiKey = generateApiKey();
114
- const config = {
115
- agentPermissions: { readContent: true, click: true, fillForms: false, scroll: true, navigate: false, apiAccess: false, automatedLogin: false, extractData: false },
116
- features: { advancedAnalytics: false, realTimeUpdates: false },
117
- restrictions: { allowedSelectors: [], blockedSelectors: ['.private', '[data-private]'], rateLimit: { maxCallsPerMinute: 60 } },
118
- logging: { enabled: false, level: 'basic' }
119
- };
120
- await pool.query(
121
- 'INSERT INTO sites (id, user_id, domain, name, description, tier, license_key, api_key, config) VALUES ($1,$2,$3,$4,$5,$6,$7,$8,$9)',
122
- [id, userId, domain, name, description || '', tier || 'free', licenseKey, apiKey, JSON.stringify(config)]
123
- );
124
- return { id, domain, name, licenseKey, apiKey, tier: tier || 'free' };
125
- }
126
-
127
- // ─── Analytics ────────────────────────────────────────────────────────
128
- async function recordAnalytic({ siteId, actionName, agentId, triggerType, success, metadata }) {
129
- await pool.query(
130
- 'INSERT INTO analytics (site_id, action_name, agent_id, trigger_type, success, metadata) VALUES ($1,$2,$3,$4,$5,$6)',
131
- [siteId, actionName, agentId || null, triggerType || null, success, JSON.stringify(metadata || {})]
132
- );
133
- }
134
-
135
- // ─── License Verification ─────────────────────────────────────────────
136
- async function verifyLicense(domain, licenseKey) {
137
- const { rows } = await pool.query(
138
- 'SELECT * FROM sites WHERE domain = $1 AND license_key = $2 AND active = TRUE', [domain, licenseKey]
139
- );
140
- const site = rows[0];
141
- if (!site) {
142
- const { rows: byKey } = await pool.query('SELECT * FROM sites WHERE license_key = $1 AND active = TRUE', [licenseKey]);
143
- if (byKey[0]) return { valid: false, error: 'Domain mismatch', tier: 'free' };
144
- return { valid: false, error: 'Invalid license key', tier: 'free' };
145
- }
146
-
147
- const tierPermissions = {
148
- free: { apiAccess: false, automatedLogin: false, extractData: false, advancedAnalytics: false },
149
- starter: { apiAccess: false, automatedLogin: true, extractData: false, advancedAnalytics: true },
150
- pro: { apiAccess: true, automatedLogin: true, extractData: true, advancedAnalytics: true },
151
- enterprise: { apiAccess: true, automatedLogin: true, extractData: true, advancedAnalytics: true }
152
- };
153
-
154
- const config = typeof site.config === 'string' ? JSON.parse(site.config) : site.config;
155
- return {
156
- valid: true,
157
- tier: site.tier,
158
- permissions: { ...config.agentPermissions, ...tierPermissions[site.tier] },
159
- restrictions: config.restrictions,
160
- features: config.features,
161
- siteId: site.id
162
- };
163
- }
164
-
165
- module.exports = {
166
- registerUser,
167
- loginUser,
168
- addSite,
169
- recordAnalytic,
170
- verifyLicense,
171
- pool
172
- };
1
+ /**
2
+ * PostgreSQL Adapter for WAB
3
+ *
4
+ * Prerequisites: npm install pg
5
+ * Set DATABASE_URL=postgres://user:pass@host:5432/wab
6
+ *
7
+ * This adapter implements the same interface as the SQLite adapter
8
+ * so it can be used as a drop-in replacement.
9
+ */
10
+
11
+ const { Pool } = require('pg');
12
+ const bcrypt = require('bcryptjs');
13
+ const { v4: uuidv4 } = require('uuid');
14
+
15
+ const pool = new Pool({ connectionString: process.env.DATABASE_URL });
16
+
17
+ // Initialize tables
18
+ async function initDB() {
19
+ await pool.query(`
20
+ CREATE TABLE IF NOT EXISTS users (
21
+ id TEXT PRIMARY KEY,
22
+ email TEXT UNIQUE NOT NULL,
23
+ password TEXT NOT NULL,
24
+ name TEXT NOT NULL,
25
+ company TEXT,
26
+ created_at TIMESTAMPTZ DEFAULT NOW(),
27
+ updated_at TIMESTAMPTZ DEFAULT NOW()
28
+ );
29
+
30
+ CREATE TABLE IF NOT EXISTS sites (
31
+ id TEXT PRIMARY KEY,
32
+ user_id TEXT NOT NULL REFERENCES users(id) ON DELETE CASCADE,
33
+ domain TEXT NOT NULL,
34
+ name TEXT NOT NULL,
35
+ description TEXT,
36
+ tier TEXT DEFAULT 'free' CHECK(tier IN ('free','starter','pro','enterprise')),
37
+ license_key TEXT UNIQUE NOT NULL,
38
+ api_key TEXT UNIQUE,
39
+ config JSONB DEFAULT '{}',
40
+ active BOOLEAN DEFAULT TRUE,
41
+ created_at TIMESTAMPTZ DEFAULT NOW(),
42
+ updated_at TIMESTAMPTZ DEFAULT NOW()
43
+ );
44
+
45
+ CREATE TABLE IF NOT EXISTS analytics (
46
+ id SERIAL PRIMARY KEY,
47
+ site_id TEXT NOT NULL REFERENCES sites(id) ON DELETE CASCADE,
48
+ action_name TEXT NOT NULL,
49
+ agent_id TEXT,
50
+ trigger_type TEXT,
51
+ success BOOLEAN,
52
+ metadata JSONB DEFAULT '{}',
53
+ created_at TIMESTAMPTZ DEFAULT NOW()
54
+ );
55
+
56
+ CREATE TABLE IF NOT EXISTS subscriptions (
57
+ id TEXT PRIMARY KEY,
58
+ user_id TEXT NOT NULL REFERENCES users(id) ON DELETE CASCADE,
59
+ site_id TEXT NOT NULL REFERENCES sites(id) ON DELETE CASCADE,
60
+ tier TEXT NOT NULL CHECK(tier IN ('free','starter','pro','enterprise')),
61
+ status TEXT DEFAULT 'active' CHECK(status IN ('active','cancelled','expired','trial')),
62
+ started_at TIMESTAMPTZ DEFAULT NOW(),
63
+ expires_at TIMESTAMPTZ
64
+ );
65
+
66
+ CREATE INDEX IF NOT EXISTS idx_sites_domain ON sites(domain);
67
+ CREATE INDEX IF NOT EXISTS idx_sites_license ON sites(license_key);
68
+ CREATE INDEX IF NOT EXISTS idx_analytics_site ON analytics(site_id);
69
+ CREATE INDEX IF NOT EXISTS idx_analytics_created ON analytics(created_at);
70
+ `);
71
+ }
72
+
73
+ initDB().catch(console.error);
74
+
75
+ function generateLicenseKey() {
76
+ const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
77
+ const segments = [];
78
+ for (let s = 0; s < 4; s++) {
79
+ let seg = '';
80
+ for (let i = 0; i < 5; i++) seg += chars[Math.floor(Math.random() * chars.length)];
81
+ segments.push(seg);
82
+ }
83
+ return `WAB-${segments.join('-')}`;
84
+ }
85
+
86
+ function generateApiKey() {
87
+ return `wab_${uuidv4().replace(/-/g, '')}`;
88
+ }
89
+
90
+ // ─── User Operations ──────────────────────────────────────────────────
91
+ async function registerUser({ email, password, name, company }) {
92
+ const id = uuidv4();
93
+ const hashed = bcrypt.hashSync(password, 12);
94
+ await pool.query(
95
+ 'INSERT INTO users (id, email, password, name, company) VALUES ($1, $2, $3, $4, $5)',
96
+ [id, email, hashed, name, company || null]
97
+ );
98
+ return { id, email, name, company };
99
+ }
100
+
101
+ async function loginUser({ email, password }) {
102
+ const { rows } = await pool.query('SELECT * FROM users WHERE email = $1', [email]);
103
+ const user = rows[0];
104
+ if (!user) return null;
105
+ if (!bcrypt.compareSync(password, user.password)) return null;
106
+ return { id: user.id, email: user.email, name: user.name, company: user.company };
107
+ }
108
+
109
+ // ─── Site Operations ──────────────────────────────────────────────────
110
+ async function addSite({ userId, domain, name, description, tier }) {
111
+ const id = uuidv4();
112
+ const licenseKey = generateLicenseKey();
113
+ const apiKey = generateApiKey();
114
+ const config = {
115
+ agentPermissions: { readContent: true, click: true, fillForms: false, scroll: true, navigate: false, apiAccess: false, automatedLogin: false, extractData: false },
116
+ features: { advancedAnalytics: false, realTimeUpdates: false },
117
+ restrictions: { allowedSelectors: [], blockedSelectors: ['.private', '[data-private]'], rateLimit: { maxCallsPerMinute: 60 } },
118
+ logging: { enabled: false, level: 'basic' }
119
+ };
120
+ await pool.query(
121
+ 'INSERT INTO sites (id, user_id, domain, name, description, tier, license_key, api_key, config) VALUES ($1,$2,$3,$4,$5,$6,$7,$8,$9)',
122
+ [id, userId, domain, name, description || '', tier || 'free', licenseKey, apiKey, JSON.stringify(config)]
123
+ );
124
+ return { id, domain, name, licenseKey, apiKey, tier: tier || 'free' };
125
+ }
126
+
127
+ // ─── Analytics ────────────────────────────────────────────────────────
128
+ async function recordAnalytic({ siteId, actionName, agentId, triggerType, success, metadata }) {
129
+ await pool.query(
130
+ 'INSERT INTO analytics (site_id, action_name, agent_id, trigger_type, success, metadata) VALUES ($1,$2,$3,$4,$5,$6)',
131
+ [siteId, actionName, agentId || null, triggerType || null, success, JSON.stringify(metadata || {})]
132
+ );
133
+ }
134
+
135
+ // ─── License Verification ─────────────────────────────────────────────
136
+ async function verifyLicense(domain, licenseKey) {
137
+ const { rows } = await pool.query(
138
+ 'SELECT * FROM sites WHERE domain = $1 AND license_key = $2 AND active = TRUE', [domain, licenseKey]
139
+ );
140
+ const site = rows[0];
141
+ if (!site) {
142
+ const { rows: byKey } = await pool.query('SELECT * FROM sites WHERE license_key = $1 AND active = TRUE', [licenseKey]);
143
+ if (byKey[0]) return { valid: false, error: 'Domain mismatch', tier: 'free' };
144
+ return { valid: false, error: 'Invalid license key', tier: 'free' };
145
+ }
146
+
147
+ const tierPermissions = {
148
+ free: { apiAccess: false, automatedLogin: false, extractData: false, advancedAnalytics: false },
149
+ starter: { apiAccess: false, automatedLogin: true, extractData: false, advancedAnalytics: true },
150
+ pro: { apiAccess: true, automatedLogin: true, extractData: true, advancedAnalytics: true },
151
+ enterprise: { apiAccess: true, automatedLogin: true, extractData: true, advancedAnalytics: true }
152
+ };
153
+
154
+ const config = typeof site.config === 'string' ? JSON.parse(site.config) : site.config;
155
+ return {
156
+ valid: true,
157
+ tier: site.tier,
158
+ permissions: { ...config.agentPermissions, ...tierPermissions[site.tier] },
159
+ restrictions: config.restrictions,
160
+ features: config.features,
161
+ siteId: site.id
162
+ };
163
+ }
164
+
165
+ module.exports = {
166
+ registerUser,
167
+ loginUser,
168
+ addSite,
169
+ recordAnalytic,
170
+ verifyLicense,
171
+ pool
172
+ };
@@ -1,7 +1,7 @@
1
- /**
2
- * SQLite Adapter — Default database backend
3
- *
4
- * Re-exports the existing db.js module as an adapter.
5
- */
6
- const db = require('../db');
7
- module.exports = db;
1
+ /**
2
+ * SQLite Adapter — Default database backend
3
+ *
4
+ * Re-exports the existing db.js module as an adapter.
5
+ */
6
+ const db = require('../db');
7
+ module.exports = db;