web-agent-bridge 3.2.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 -72
- package/README.ar.md +1286 -1152
- package/README.md +1764 -1635
- 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 +16 -3
- 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 -0
- package/public/agent-workspace.html +349 -349
- package/public/ai.html +198 -198
- package/public/api.html +413 -412
- package/public/browser.html +486 -486
- package/public/commander-dashboard.html +243 -243
- package/public/cookies.html +210 -210
- 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 -706
- package/public/dns.html +507 -0
- package/public/docs.html +587 -587
- package/public/feed.xml +89 -89
- package/public/growth.html +463 -463
- package/public/index.html +1070 -982
- 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 -360
- package/public/llms.txt +125 -125
- package/public/login.html +85 -85
- 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 -2489
- package/public/premium.html +793 -793
- package/public/privacy.html +297 -297
- package/public/register.html +105 -105
- 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 -256
- 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 +12 -1
- package/sdk/multi-agent.js +318 -318
- package/sdk/package.json +1 -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 +531 -427
- 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 -540
- 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 -332
- package/server/routes/demo-store.js +154 -0
- package/server/routes/discovery.js +417 -417
- package/server/routes/gateway.js +173 -157
- 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 -185
- package/server/routes/wab-api.js +850 -501
- 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 -261
- 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 -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/plugins.js +771 -771
- package/server/services/price-intelligence.js +566 -566
- 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/public/score.html +0 -263
- package/server/migrations/006_growth_suite.sql +0 -138
- package/server/routes/growth.js +0 -962
- package/server/services/fairness-engine.js +0 -409
- package/server/services/fairness.js +0 -420
package/server/utils/cache.js
CHANGED
|
@@ -1,125 +1,125 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* WAB Caching Layer — In-memory cache with TTL for hot data
|
|
3
|
-
* Reduces DB reads for license verification, config, and stats
|
|
4
|
-
*/
|
|
5
|
-
class Cache {
|
|
6
|
-
constructor(defaultTTL = 60000) {
|
|
7
|
-
this.store = new Map();
|
|
8
|
-
this.defaultTTL = defaultTTL;
|
|
9
|
-
this._interval = setInterval(() => this._cleanup(), 120000);
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
get(key) {
|
|
13
|
-
const entry = this.store.get(key);
|
|
14
|
-
if (!entry) return undefined;
|
|
15
|
-
if (Date.now() > entry.expiresAt) {
|
|
16
|
-
this.store.delete(key);
|
|
17
|
-
return undefined;
|
|
18
|
-
}
|
|
19
|
-
entry.hits++;
|
|
20
|
-
return entry.value;
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
set(key, value, ttl) {
|
|
24
|
-
this.store.set(key, {
|
|
25
|
-
value,
|
|
26
|
-
expiresAt: Date.now() + (ttl || this.defaultTTL),
|
|
27
|
-
hits: 0
|
|
28
|
-
});
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
del(key) {
|
|
32
|
-
this.store.delete(key);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
invalidatePattern(pattern) {
|
|
36
|
-
for (const key of this.store.keys()) {
|
|
37
|
-
if (key.includes(pattern)) this.store.delete(key);
|
|
38
|
-
}
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
stats() {
|
|
42
|
-
return {
|
|
43
|
-
size: this.store.size,
|
|
44
|
-
keys: Array.from(this.store.keys())
|
|
45
|
-
};
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
_cleanup() {
|
|
49
|
-
const now = Date.now();
|
|
50
|
-
for (const [key, entry] of this.store) {
|
|
51
|
-
if (now > entry.expiresAt) this.store.delete(key);
|
|
52
|
-
}
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
destroy() {
|
|
56
|
-
clearInterval(this._interval);
|
|
57
|
-
this.store.clear();
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
/**
|
|
62
|
-
* Analytics Queue — Batches analytics inserts for better write performance
|
|
63
|
-
* Flushes every N seconds or when buffer reaches max size.
|
|
64
|
-
* maxBufferTotal caps memory if DB writes fail repeatedly (DoS mitigation).
|
|
65
|
-
*/
|
|
66
|
-
class AnalyticsQueue {
|
|
67
|
-
constructor(db, options = {}) {
|
|
68
|
-
this.db = db;
|
|
69
|
-
this.buffer = [];
|
|
70
|
-
this.maxSize = options.maxSize || 50;
|
|
71
|
-
this.maxBufferTotal = options.maxBufferTotal || 5000;
|
|
72
|
-
this.flushInterval = options.flushInterval || 5000;
|
|
73
|
-
this._timer = setInterval(() => this.flush(), this.flushInterval);
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
push(analytic) {
|
|
77
|
-
while (this.buffer.length >= this.maxBufferTotal) {
|
|
78
|
-
this.buffer.shift();
|
|
79
|
-
console.warn('[WAB] Analytics buffer at cap; dropped oldest event');
|
|
80
|
-
}
|
|
81
|
-
this.buffer.push(analytic);
|
|
82
|
-
if (this.buffer.length >= this.maxSize) {
|
|
83
|
-
this.flush();
|
|
84
|
-
}
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
flush() {
|
|
88
|
-
if (this.buffer.length === 0) return;
|
|
89
|
-
const batch = this.buffer.splice(0);
|
|
90
|
-
|
|
91
|
-
try {
|
|
92
|
-
const insert = this.db.prepare(
|
|
93
|
-
`INSERT INTO analytics (site_id, action_name, agent_id, trigger_type, success, metadata) VALUES (?, ?, ?, ?, ?, ?)`
|
|
94
|
-
);
|
|
95
|
-
const insertMany = this.db.transaction((items) => {
|
|
96
|
-
for (const item of items) {
|
|
97
|
-
insert.run(
|
|
98
|
-
item.siteId,
|
|
99
|
-
item.actionName,
|
|
100
|
-
item.agentId || null,
|
|
101
|
-
item.triggerType || null,
|
|
102
|
-
item.success ? 1 : 0,
|
|
103
|
-
JSON.stringify(item.metadata || {})
|
|
104
|
-
);
|
|
105
|
-
}
|
|
106
|
-
});
|
|
107
|
-
insertMany(batch);
|
|
108
|
-
} catch (err) {
|
|
109
|
-
console.error('[WAB Cache] Analytics batch insert failed:', err.message);
|
|
110
|
-
while (this.buffer.length + batch.length > this.maxBufferTotal) {
|
|
111
|
-
batch.shift();
|
|
112
|
-
}
|
|
113
|
-
this.buffer.unshift(...batch);
|
|
114
|
-
}
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
destroy() {
|
|
118
|
-
clearInterval(this._timer);
|
|
119
|
-
this.flush();
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
const cache = new Cache(60000);
|
|
124
|
-
|
|
125
|
-
module.exports = { Cache, AnalyticsQueue, cache };
|
|
1
|
+
/**
|
|
2
|
+
* WAB Caching Layer — In-memory cache with TTL for hot data
|
|
3
|
+
* Reduces DB reads for license verification, config, and stats
|
|
4
|
+
*/
|
|
5
|
+
class Cache {
|
|
6
|
+
constructor(defaultTTL = 60000) {
|
|
7
|
+
this.store = new Map();
|
|
8
|
+
this.defaultTTL = defaultTTL;
|
|
9
|
+
this._interval = setInterval(() => this._cleanup(), 120000);
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
get(key) {
|
|
13
|
+
const entry = this.store.get(key);
|
|
14
|
+
if (!entry) return undefined;
|
|
15
|
+
if (Date.now() > entry.expiresAt) {
|
|
16
|
+
this.store.delete(key);
|
|
17
|
+
return undefined;
|
|
18
|
+
}
|
|
19
|
+
entry.hits++;
|
|
20
|
+
return entry.value;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
set(key, value, ttl) {
|
|
24
|
+
this.store.set(key, {
|
|
25
|
+
value,
|
|
26
|
+
expiresAt: Date.now() + (ttl || this.defaultTTL),
|
|
27
|
+
hits: 0
|
|
28
|
+
});
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
del(key) {
|
|
32
|
+
this.store.delete(key);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
invalidatePattern(pattern) {
|
|
36
|
+
for (const key of this.store.keys()) {
|
|
37
|
+
if (key.includes(pattern)) this.store.delete(key);
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
stats() {
|
|
42
|
+
return {
|
|
43
|
+
size: this.store.size,
|
|
44
|
+
keys: Array.from(this.store.keys())
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
_cleanup() {
|
|
49
|
+
const now = Date.now();
|
|
50
|
+
for (const [key, entry] of this.store) {
|
|
51
|
+
if (now > entry.expiresAt) this.store.delete(key);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
destroy() {
|
|
56
|
+
clearInterval(this._interval);
|
|
57
|
+
this.store.clear();
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Analytics Queue — Batches analytics inserts for better write performance
|
|
63
|
+
* Flushes every N seconds or when buffer reaches max size.
|
|
64
|
+
* maxBufferTotal caps memory if DB writes fail repeatedly (DoS mitigation).
|
|
65
|
+
*/
|
|
66
|
+
class AnalyticsQueue {
|
|
67
|
+
constructor(db, options = {}) {
|
|
68
|
+
this.db = db;
|
|
69
|
+
this.buffer = [];
|
|
70
|
+
this.maxSize = options.maxSize || 50;
|
|
71
|
+
this.maxBufferTotal = options.maxBufferTotal || 5000;
|
|
72
|
+
this.flushInterval = options.flushInterval || 5000;
|
|
73
|
+
this._timer = setInterval(() => this.flush(), this.flushInterval);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
push(analytic) {
|
|
77
|
+
while (this.buffer.length >= this.maxBufferTotal) {
|
|
78
|
+
this.buffer.shift();
|
|
79
|
+
console.warn('[WAB] Analytics buffer at cap; dropped oldest event');
|
|
80
|
+
}
|
|
81
|
+
this.buffer.push(analytic);
|
|
82
|
+
if (this.buffer.length >= this.maxSize) {
|
|
83
|
+
this.flush();
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
flush() {
|
|
88
|
+
if (this.buffer.length === 0) return;
|
|
89
|
+
const batch = this.buffer.splice(0);
|
|
90
|
+
|
|
91
|
+
try {
|
|
92
|
+
const insert = this.db.prepare(
|
|
93
|
+
`INSERT INTO analytics (site_id, action_name, agent_id, trigger_type, success, metadata) VALUES (?, ?, ?, ?, ?, ?)`
|
|
94
|
+
);
|
|
95
|
+
const insertMany = this.db.transaction((items) => {
|
|
96
|
+
for (const item of items) {
|
|
97
|
+
insert.run(
|
|
98
|
+
item.siteId,
|
|
99
|
+
item.actionName,
|
|
100
|
+
item.agentId || null,
|
|
101
|
+
item.triggerType || null,
|
|
102
|
+
item.success ? 1 : 0,
|
|
103
|
+
JSON.stringify(item.metadata || {})
|
|
104
|
+
);
|
|
105
|
+
}
|
|
106
|
+
});
|
|
107
|
+
insertMany(batch);
|
|
108
|
+
} catch (err) {
|
|
109
|
+
console.error('[WAB Cache] Analytics batch insert failed:', err.message);
|
|
110
|
+
while (this.buffer.length + batch.length > this.maxBufferTotal) {
|
|
111
|
+
batch.shift();
|
|
112
|
+
}
|
|
113
|
+
this.buffer.unshift(...batch);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
destroy() {
|
|
118
|
+
clearInterval(this._timer);
|
|
119
|
+
this.flush();
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
const cache = new Cache(60000);
|
|
124
|
+
|
|
125
|
+
module.exports = { Cache, AnalyticsQueue, cache };
|
package/server/utils/migrate.js
CHANGED
|
@@ -1,81 +1,81 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Database Migration Runner
|
|
3
|
-
* Tracks and applies SQL migrations from server/migrations/ in order.
|
|
4
|
-
* Uses a `_migrations` table to record applied migrations.
|
|
5
|
-
*/
|
|
6
|
-
const Database = require('better-sqlite3');
|
|
7
|
-
const path = require('path');
|
|
8
|
-
const fs = require('fs');
|
|
9
|
-
|
|
10
|
-
const DATA_DIR = process.env.NODE_ENV === 'test'
|
|
11
|
-
? path.join(__dirname, '..', '..', 'data-test')
|
|
12
|
-
: path.join(__dirname, '..', '..', 'data');
|
|
13
|
-
if (!fs.existsSync(DATA_DIR)) fs.mkdirSync(DATA_DIR, { recursive: true });
|
|
14
|
-
|
|
15
|
-
const dbFile = process.env.NODE_ENV === 'test' ? 'wab-test.db' : 'wab.db';
|
|
16
|
-
const db = new Database(path.join(DATA_DIR, dbFile));
|
|
17
|
-
|
|
18
|
-
// Ensure migrations tracking table exists
|
|
19
|
-
db.exec(`
|
|
20
|
-
CREATE TABLE IF NOT EXISTS _migrations (
|
|
21
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
22
|
-
name TEXT UNIQUE NOT NULL,
|
|
23
|
-
applied_at TEXT DEFAULT (datetime('now'))
|
|
24
|
-
);
|
|
25
|
-
`);
|
|
26
|
-
|
|
27
|
-
const MIGRATIONS_DIR = path.join(__dirname, '..', 'migrations');
|
|
28
|
-
|
|
29
|
-
function getAppliedMigrations() {
|
|
30
|
-
return new Set(
|
|
31
|
-
db.prepare('SELECT name FROM _migrations ORDER BY id').all().map(r => r.name)
|
|
32
|
-
);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
function runMigrations() {
|
|
36
|
-
if (!fs.existsSync(MIGRATIONS_DIR)) {
|
|
37
|
-
console.log('No migrations directory found.');
|
|
38
|
-
return;
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
const files = fs.readdirSync(MIGRATIONS_DIR)
|
|
42
|
-
.filter(f => f.endsWith('.sql'))
|
|
43
|
-
.sort();
|
|
44
|
-
|
|
45
|
-
const applied = getAppliedMigrations();
|
|
46
|
-
let count = 0;
|
|
47
|
-
|
|
48
|
-
const applyMigration = db.transaction((name, sql) => {
|
|
49
|
-
db.exec(sql);
|
|
50
|
-
db.prepare('INSERT INTO _migrations (name) VALUES (?)').run(name);
|
|
51
|
-
});
|
|
52
|
-
|
|
53
|
-
for (const file of files) {
|
|
54
|
-
if (applied.has(file)) continue;
|
|
55
|
-
|
|
56
|
-
const sql = fs.readFileSync(path.join(MIGRATIONS_DIR, file), 'utf8');
|
|
57
|
-
try {
|
|
58
|
-
applyMigration(file, sql);
|
|
59
|
-
console.log(` ✅ Migration applied: ${file}`);
|
|
60
|
-
count++;
|
|
61
|
-
} catch (err) {
|
|
62
|
-
console.error(` ❌ Migration failed: ${file} — ${err.message}`);
|
|
63
|
-
process.exit(1);
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
if (count === 0) {
|
|
68
|
-
console.log(' All migrations up to date.');
|
|
69
|
-
} else {
|
|
70
|
-
console.log(` ${count} migration(s) applied.`);
|
|
71
|
-
}
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
// Run when called directly: node server/utils/migrate.js
|
|
75
|
-
if (require.main === module) {
|
|
76
|
-
console.log('Running database migrations...');
|
|
77
|
-
runMigrations();
|
|
78
|
-
db.close();
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
module.exports = { runMigrations };
|
|
1
|
+
/**
|
|
2
|
+
* Database Migration Runner
|
|
3
|
+
* Tracks and applies SQL migrations from server/migrations/ in order.
|
|
4
|
+
* Uses a `_migrations` table to record applied migrations.
|
|
5
|
+
*/
|
|
6
|
+
const Database = require('better-sqlite3');
|
|
7
|
+
const path = require('path');
|
|
8
|
+
const fs = require('fs');
|
|
9
|
+
|
|
10
|
+
const DATA_DIR = process.env.NODE_ENV === 'test'
|
|
11
|
+
? path.join(__dirname, '..', '..', 'data-test')
|
|
12
|
+
: (process.env.DATA_DIR || path.join(__dirname, '..', '..', 'data'));
|
|
13
|
+
if (!fs.existsSync(DATA_DIR)) fs.mkdirSync(DATA_DIR, { recursive: true });
|
|
14
|
+
|
|
15
|
+
const dbFile = process.env.NODE_ENV === 'test' ? 'wab-test.db' : 'wab.db';
|
|
16
|
+
const db = new Database(path.join(DATA_DIR, dbFile));
|
|
17
|
+
|
|
18
|
+
// Ensure migrations tracking table exists
|
|
19
|
+
db.exec(`
|
|
20
|
+
CREATE TABLE IF NOT EXISTS _migrations (
|
|
21
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
22
|
+
name TEXT UNIQUE NOT NULL,
|
|
23
|
+
applied_at TEXT DEFAULT (datetime('now'))
|
|
24
|
+
);
|
|
25
|
+
`);
|
|
26
|
+
|
|
27
|
+
const MIGRATIONS_DIR = path.join(__dirname, '..', 'migrations');
|
|
28
|
+
|
|
29
|
+
function getAppliedMigrations() {
|
|
30
|
+
return new Set(
|
|
31
|
+
db.prepare('SELECT name FROM _migrations ORDER BY id').all().map(r => r.name)
|
|
32
|
+
);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
function runMigrations() {
|
|
36
|
+
if (!fs.existsSync(MIGRATIONS_DIR)) {
|
|
37
|
+
console.log('No migrations directory found.');
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
const files = fs.readdirSync(MIGRATIONS_DIR)
|
|
42
|
+
.filter(f => f.endsWith('.sql'))
|
|
43
|
+
.sort();
|
|
44
|
+
|
|
45
|
+
const applied = getAppliedMigrations();
|
|
46
|
+
let count = 0;
|
|
47
|
+
|
|
48
|
+
const applyMigration = db.transaction((name, sql) => {
|
|
49
|
+
db.exec(sql);
|
|
50
|
+
db.prepare('INSERT INTO _migrations (name) VALUES (?)').run(name);
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
for (const file of files) {
|
|
54
|
+
if (applied.has(file)) continue;
|
|
55
|
+
|
|
56
|
+
const sql = fs.readFileSync(path.join(MIGRATIONS_DIR, file), 'utf8');
|
|
57
|
+
try {
|
|
58
|
+
applyMigration(file, sql);
|
|
59
|
+
console.log(` ✅ Migration applied: ${file}`);
|
|
60
|
+
count++;
|
|
61
|
+
} catch (err) {
|
|
62
|
+
console.error(` ❌ Migration failed: ${file} — ${err.message}`);
|
|
63
|
+
process.exit(1);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
if (count === 0) {
|
|
68
|
+
console.log(' All migrations up to date.');
|
|
69
|
+
} else {
|
|
70
|
+
console.log(` ${count} migration(s) applied.`);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// Run when called directly: node server/utils/migrate.js
|
|
75
|
+
if (require.main === module) {
|
|
76
|
+
console.log('Running database migrations...');
|
|
77
|
+
runMigrations();
|
|
78
|
+
db.close();
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
module.exports = { runMigrations };
|
|
@@ -0,0 +1,228 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Safe Fetch — SSRF-resistant HTTP client.
|
|
5
|
+
*
|
|
6
|
+
* Mitigations applied:
|
|
7
|
+
* 1. Scheme allow-list (http/https only).
|
|
8
|
+
* 2. Optional domain allow-list (string globs or "*").
|
|
9
|
+
* 3. DNS resolution + private/reserved/loopback/link-local CIDR block.
|
|
10
|
+
* 4. Re-validation on EVERY redirect hop (manual redirect handling).
|
|
11
|
+
* 5. Hard timeout via AbortController.
|
|
12
|
+
* 6. Max response body size (default 5 MB) — drains and aborts.
|
|
13
|
+
* 7. Optional Content-Type allow-list.
|
|
14
|
+
*
|
|
15
|
+
* NEVER call native `fetch(url)` directly with user-supplied URLs anywhere
|
|
16
|
+
* inside this server process. Use this helper.
|
|
17
|
+
*/
|
|
18
|
+
|
|
19
|
+
const dns = require('node:dns').promises;
|
|
20
|
+
const net = require('node:net');
|
|
21
|
+
|
|
22
|
+
const PRIVATE_V4_CIDRS = [
|
|
23
|
+
['10.0.0.0', 8],
|
|
24
|
+
['172.16.0.0', 12],
|
|
25
|
+
['192.168.0.0', 16],
|
|
26
|
+
['127.0.0.0', 8],
|
|
27
|
+
['169.254.0.0', 16], // link-local (AWS metadata 169.254.169.254 lives here)
|
|
28
|
+
['100.64.0.0', 10], // CGNAT
|
|
29
|
+
['0.0.0.0', 8],
|
|
30
|
+
['224.0.0.0', 4], // multicast
|
|
31
|
+
['240.0.0.0', 4], // reserved
|
|
32
|
+
['198.18.0.0', 15], // benchmarking
|
|
33
|
+
['192.0.0.0', 24],
|
|
34
|
+
['192.0.2.0', 24], // TEST-NET-1
|
|
35
|
+
['198.51.100.0', 24], // TEST-NET-2
|
|
36
|
+
['203.0.113.0', 24], // TEST-NET-3
|
|
37
|
+
];
|
|
38
|
+
|
|
39
|
+
const PRIVATE_V6_PREFIXES = ['::1', 'fc', 'fd', 'fe80', 'ff', '::ffff:127.', '::ffff:10.', '::ffff:192.168.', '::ffff:172.', '::', '64:ff9b::'];
|
|
40
|
+
|
|
41
|
+
function _ipToInt(ip) {
|
|
42
|
+
return ip.split('.').reduce((acc, oct) => (acc << 8) + parseInt(oct, 10), 0) >>> 0;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
function _isPrivateV4(ip) {
|
|
46
|
+
const ipInt = _ipToInt(ip);
|
|
47
|
+
return PRIVATE_V4_CIDRS.some(([base, bits]) => {
|
|
48
|
+
const baseInt = _ipToInt(base);
|
|
49
|
+
const mask = bits === 0 ? 0 : (~0 << (32 - bits)) >>> 0;
|
|
50
|
+
return (ipInt & mask) === (baseInt & mask);
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
function _isPrivateV6(ip) {
|
|
55
|
+
const lower = ip.toLowerCase();
|
|
56
|
+
return PRIVATE_V6_PREFIXES.some((p) => lower === p || lower.startsWith(p));
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
function isPrivateAddress(ip) {
|
|
60
|
+
if (!ip) return true;
|
|
61
|
+
if (net.isIPv4(ip)) return _isPrivateV4(ip);
|
|
62
|
+
if (net.isIPv6(ip)) return _isPrivateV6(ip);
|
|
63
|
+
return true; // unknown → treat as private
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
function _matchesGlob(host, pattern) {
|
|
67
|
+
if (pattern === '*') return true;
|
|
68
|
+
if (pattern.startsWith('*.')) {
|
|
69
|
+
const suffix = pattern.slice(2).toLowerCase();
|
|
70
|
+
return host === suffix || host.endsWith('.' + suffix);
|
|
71
|
+
}
|
|
72
|
+
return host.toLowerCase() === pattern.toLowerCase();
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
function _allowedHost(host, allowList) {
|
|
76
|
+
if (!allowList || allowList.length === 0) return true; // no list = allow public
|
|
77
|
+
return allowList.some((p) => _matchesGlob(host, p));
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
async function _assertSafeHost(hostname) {
|
|
81
|
+
// If it's already an IP literal, validate directly.
|
|
82
|
+
if (net.isIP(hostname)) {
|
|
83
|
+
if (isPrivateAddress(hostname)) {
|
|
84
|
+
throw new Error(`SSRF blocked: private/reserved IP ${hostname}`);
|
|
85
|
+
}
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
let records;
|
|
89
|
+
try {
|
|
90
|
+
records = await dns.lookup(hostname, { all: true, verbatim: true });
|
|
91
|
+
} catch (err) {
|
|
92
|
+
throw new Error(`DNS resolution failed for ${hostname}: ${err.code || err.message}`);
|
|
93
|
+
}
|
|
94
|
+
if (!records || records.length === 0) {
|
|
95
|
+
throw new Error(`DNS returned no records for ${hostname}`);
|
|
96
|
+
}
|
|
97
|
+
for (const r of records) {
|
|
98
|
+
if (isPrivateAddress(r.address)) {
|
|
99
|
+
throw new Error(`SSRF blocked: ${hostname} resolves to private/reserved address ${r.address}`);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
/**
|
|
105
|
+
* Validate a URL string against the SSRF policy.
|
|
106
|
+
* @returns {URL} parsed URL
|
|
107
|
+
*/
|
|
108
|
+
async function validateUrl(rawUrl, options = {}) {
|
|
109
|
+
let parsed;
|
|
110
|
+
try {
|
|
111
|
+
parsed = new URL(rawUrl);
|
|
112
|
+
} catch (e) {
|
|
113
|
+
throw new Error('Invalid URL');
|
|
114
|
+
}
|
|
115
|
+
const allowedSchemes = options.allowedSchemes || ['http:', 'https:'];
|
|
116
|
+
if (!allowedSchemes.includes(parsed.protocol)) {
|
|
117
|
+
throw new Error(`Scheme ${parsed.protocol} not allowed`);
|
|
118
|
+
}
|
|
119
|
+
if (options.requireHttps && parsed.protocol !== 'https:') {
|
|
120
|
+
throw new Error('HTTPS required');
|
|
121
|
+
}
|
|
122
|
+
// Block credentials in URLs (defeats some auth-smuggling attacks).
|
|
123
|
+
if (parsed.username || parsed.password) {
|
|
124
|
+
throw new Error('Credentials in URLs are not allowed');
|
|
125
|
+
}
|
|
126
|
+
// Restrict ports to defaults unless explicitly allowed.
|
|
127
|
+
const allowedPorts = options.allowedPorts || [80, 443];
|
|
128
|
+
const port = parsed.port ? parseInt(parsed.port, 10) : (parsed.protocol === 'https:' ? 443 : 80);
|
|
129
|
+
if (!allowedPorts.includes(port)) {
|
|
130
|
+
throw new Error(`Port ${port} not allowed`);
|
|
131
|
+
}
|
|
132
|
+
if (!_allowedHost(parsed.hostname, options.allowList)) {
|
|
133
|
+
throw new Error(`Host ${parsed.hostname} not in allow-list`);
|
|
134
|
+
}
|
|
135
|
+
await _assertSafeHost(parsed.hostname);
|
|
136
|
+
return parsed;
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
/**
|
|
140
|
+
* SSRF-resistant fetch.
|
|
141
|
+
* Manually follows redirects so each hop is re-validated.
|
|
142
|
+
*/
|
|
143
|
+
async function safeFetch(rawUrl, init = {}, opts = {}) {
|
|
144
|
+
const maxRedirects = opts.maxRedirects ?? 3;
|
|
145
|
+
const maxBytes = opts.maxBytes ?? 5 * 1024 * 1024; // 5 MB
|
|
146
|
+
const timeoutMs = opts.timeoutMs ?? 10000;
|
|
147
|
+
const allowList = opts.allowList;
|
|
148
|
+
const allowedContentTypes = opts.allowedContentTypes; // e.g. ['text/html','application/json']
|
|
149
|
+
|
|
150
|
+
let currentUrl = rawUrl;
|
|
151
|
+
for (let hop = 0; hop <= maxRedirects; hop++) {
|
|
152
|
+
await validateUrl(currentUrl, { allowList, allowedSchemes: opts.allowedSchemes, requireHttps: opts.requireHttps, allowedPorts: opts.allowedPorts });
|
|
153
|
+
|
|
154
|
+
const controller = new AbortController();
|
|
155
|
+
const timer = setTimeout(() => controller.abort(), timeoutMs);
|
|
156
|
+
|
|
157
|
+
let res;
|
|
158
|
+
try {
|
|
159
|
+
res = await fetch(currentUrl, {
|
|
160
|
+
...init,
|
|
161
|
+
redirect: 'manual',
|
|
162
|
+
signal: controller.signal,
|
|
163
|
+
});
|
|
164
|
+
} finally {
|
|
165
|
+
clearTimeout(timer);
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
if (res.status >= 300 && res.status < 400) {
|
|
169
|
+
const loc = res.headers.get('location');
|
|
170
|
+
if (!loc) return _enforceBody(res, maxBytes, allowedContentTypes);
|
|
171
|
+
if (hop === maxRedirects) throw new Error('Too many redirects');
|
|
172
|
+
currentUrl = new URL(loc, currentUrl).toString();
|
|
173
|
+
continue;
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
return _enforceBody(res, maxBytes, allowedContentTypes);
|
|
177
|
+
}
|
|
178
|
+
throw new Error('Redirect loop');
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
async function _enforceBody(res, maxBytes, allowedContentTypes) {
|
|
182
|
+
const ct = (res.headers.get('content-type') || '').toLowerCase();
|
|
183
|
+
if (allowedContentTypes && !allowedContentTypes.some((t) => ct.startsWith(t))) {
|
|
184
|
+
res.body?.cancel?.().catch(() => {});
|
|
185
|
+
throw new Error(`Content-Type ${ct || 'unknown'} not allowed`);
|
|
186
|
+
}
|
|
187
|
+
const declared = parseInt(res.headers.get('content-length') || '0', 10);
|
|
188
|
+
if (declared && declared > maxBytes) {
|
|
189
|
+
res.body?.cancel?.().catch(() => {});
|
|
190
|
+
throw new Error(`Response too large (${declared} bytes, max ${maxBytes})`);
|
|
191
|
+
}
|
|
192
|
+
if (!res.body) return res;
|
|
193
|
+
|
|
194
|
+
// Buffer the response while enforcing size.
|
|
195
|
+
const reader = res.body.getReader();
|
|
196
|
+
const chunks = [];
|
|
197
|
+
let total = 0;
|
|
198
|
+
while (true) {
|
|
199
|
+
const { done, value } = await reader.read();
|
|
200
|
+
if (done) break;
|
|
201
|
+
total += value.byteLength;
|
|
202
|
+
if (total > maxBytes) {
|
|
203
|
+
reader.cancel().catch(() => {});
|
|
204
|
+
throw new Error(`Response exceeded ${maxBytes} bytes`);
|
|
205
|
+
}
|
|
206
|
+
chunks.push(value);
|
|
207
|
+
}
|
|
208
|
+
const buf = Buffer.concat(chunks.map((c) => Buffer.from(c)));
|
|
209
|
+
// Return a Response-like wrapper matching the parts callers actually use.
|
|
210
|
+
return {
|
|
211
|
+
ok: res.ok,
|
|
212
|
+
status: res.status,
|
|
213
|
+
statusText: res.statusText,
|
|
214
|
+
headers: res.headers,
|
|
215
|
+
url: res.url,
|
|
216
|
+
redirected: res.redirected,
|
|
217
|
+
async text() { return buf.toString('utf8'); },
|
|
218
|
+
async json() { return JSON.parse(buf.toString('utf8')); },
|
|
219
|
+
async arrayBuffer() { return buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength); },
|
|
220
|
+
async buffer() { return buf; },
|
|
221
|
+
};
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
module.exports = {
|
|
225
|
+
safeFetch,
|
|
226
|
+
validateUrl,
|
|
227
|
+
isPrivateAddress,
|
|
228
|
+
};
|