web-agent-bridge 1.1.1 → 1.1.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (66) hide show
  1. package/LICENSE +21 -21
  2. package/README.ar.md +446 -446
  3. package/README.md +844 -844
  4. package/bin/cli.js +80 -80
  5. package/bin/wab.js +80 -80
  6. package/docs/DEPLOY.md +118 -118
  7. package/docs/SPEC.md +1540 -1540
  8. package/examples/bidi-agent.js +119 -119
  9. package/examples/mcp-agent.js +94 -94
  10. package/examples/puppeteer-agent.js +108 -108
  11. package/examples/vision-agent.js +171 -171
  12. package/package.json +78 -78
  13. package/public/admin/dashboard.html +848 -848
  14. package/public/admin/login.html +84 -84
  15. package/public/cookies.html +208 -208
  16. package/public/css/styles.css +1235 -1235
  17. package/public/dashboard.html +704 -704
  18. package/public/docs.html +585 -585
  19. package/public/index.html +332 -332
  20. package/public/js/auth-nav.js +31 -31
  21. package/public/js/auth-redirect.js +12 -12
  22. package/public/js/cookie-consent.js +56 -56
  23. package/public/js/ws-client.js +74 -74
  24. package/public/login.html +83 -83
  25. package/public/privacy.html +295 -295
  26. package/public/register.html +103 -103
  27. package/public/terms.html +254 -254
  28. package/script/ai-agent-bridge.js +1513 -1513
  29. package/sdk/README.md +55 -55
  30. package/sdk/index.js +203 -203
  31. package/sdk/package.json +14 -14
  32. package/server/config/secrets.js +92 -92
  33. package/server/index.js +181 -181
  34. package/server/middleware/adminAuth.js +30 -30
  35. package/server/middleware/auth.js +41 -41
  36. package/server/middleware/rateLimits.js +24 -24
  37. package/server/migrations/001_add_analytics_indexes.sql +7 -7
  38. package/server/models/adapters/index.js +33 -33
  39. package/server/models/adapters/mysql.js +183 -183
  40. package/server/models/adapters/postgresql.js +172 -172
  41. package/server/models/adapters/sqlite.js +7 -7
  42. package/server/models/db.js +561 -561
  43. package/server/routes/admin.js +247 -247
  44. package/server/routes/api.js +138 -138
  45. package/server/routes/auth.js +51 -51
  46. package/server/routes/billing.js +45 -45
  47. package/server/routes/discovery.js +329 -329
  48. package/server/routes/license.js +240 -240
  49. package/server/routes/noscript.js +543 -543
  50. package/server/routes/wab-api.js +476 -476
  51. package/server/services/email.js +204 -204
  52. package/server/services/fairness.js +420 -420
  53. package/server/services/stripe.js +192 -192
  54. package/server/utils/cache.js +125 -125
  55. package/server/utils/migrate.js +81 -81
  56. package/server/utils/secureFields.js +50 -50
  57. package/server/ws.js +101 -101
  58. package/wab-mcp-adapter/README.md +136 -136
  59. package/wab-mcp-adapter/index.js +555 -555
  60. package/wab-mcp-adapter/package.json +17 -17
  61. package/public/css/premium.css +0 -317
  62. package/public/premium-dashboard.html +0 -2075
  63. package/public/premium.html +0 -791
  64. package/server/migrations/002_premium_features.sql +0 -418
  65. package/server/routes/premium.js +0 -724
  66. package/server/services/premium.js +0 -1680
@@ -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;