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.
- package/LICENSE +21 -21
- package/README.ar.md +446 -446
- package/README.md +780 -844
- package/bin/cli.js +80 -80
- package/bin/wab.js +80 -80
- package/examples/bidi-agent.js +119 -119
- package/examples/mcp-agent.js +94 -94
- package/examples/next-app-router/README.md +44 -0
- package/examples/puppeteer-agent.js +108 -108
- package/examples/saas-dashboard/README.md +55 -0
- package/examples/shopify-hydrogen/README.md +74 -0
- package/examples/vision-agent.js +171 -171
- package/examples/wordpress-elementor/README.md +77 -0
- package/package.json +69 -78
- package/public/.well-known/ai-assets.json +59 -0
- package/public/admin/login.html +84 -84
- package/public/ai.html +196 -0
- package/public/cookies.html +208 -208
- package/public/css/premium.css +317 -0
- package/public/css/styles.css +1235 -1235
- package/public/dashboard.html +704 -704
- package/public/demo.html +259 -0
- package/public/docs.html +585 -585
- package/public/feed.xml +89 -0
- package/public/index.html +495 -332
- 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 -0
- package/public/js/ws-client.js +74 -74
- package/public/llms-full.txt +309 -0
- package/public/llms.txt +85 -0
- package/public/login.html +83 -83
- package/public/openapi.json +580 -0
- package/public/premium-dashboard.html +2487 -0
- package/public/premium.html +791 -0
- package/public/privacy.html +295 -295
- package/public/register.html +103 -103
- package/public/robots.txt +87 -0
- package/public/script/wab-consent.d.ts +36 -0
- package/public/script/wab-consent.js +104 -0
- package/public/script/wab-schema.js +131 -0
- package/public/script/wab.d.ts +108 -0
- package/public/script/wab.min.js +234 -0
- package/public/sitemap.xml +93 -0
- package/public/terms.html +254 -254
- package/public/video/tutorial.mp4 +0 -0
- package/script/ai-agent-bridge.js +1558 -1513
- package/sdk/README.md +55 -55
- package/sdk/index.d.ts +118 -0
- package/sdk/index.js +257 -203
- package/sdk/package.json +14 -14
- package/sdk/schema-discovery.js +83 -0
- package/server/config/secrets.js +94 -92
- package/server/index.js +0 -9
- package/server/middleware/adminAuth.js +30 -30
- package/server/middleware/auth.js +41 -41
- package/server/middleware/rateLimits.js +24 -24
- package/server/migrations/001_add_analytics_indexes.sql +7 -7
- package/server/migrations/002_premium_features.sql +418 -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 +561 -561
- package/server/routes/admin-premium.js +671 -0
- package/server/routes/admin.js +247 -247
- package/server/routes/api.js +131 -138
- package/server/routes/auth.js +51 -51
- package/server/routes/billing.js +45 -45
- package/server/routes/discovery.js +406 -329
- package/server/routes/license.js +240 -240
- package/server/routes/noscript.js +543 -543
- package/server/routes/premium-v2.js +686 -0
- package/server/routes/premium.js +724 -0
- package/server/routes/wab-api.js +476 -476
- package/server/services/agent-memory.js +625 -0
- package/server/services/email.js +204 -204
- package/server/services/fairness.js +420 -420
- package/server/services/plugins.js +747 -0
- package/server/services/premium.js +1883 -0
- package/server/services/self-healing.js +843 -0
- package/server/services/stripe.js +192 -192
- package/server/services/swarm.js +788 -0
- package/server/services/vision.js +871 -0
- package/server/utils/cache.js +125 -125
- package/server/utils/migrate.js +81 -81
- package/server/utils/secureFields.js +50 -50
- package/server/ws.js +101 -101
- package/docs/DEPLOY.md +0 -118
- package/docs/SPEC.md +0 -1540
- package/wab-mcp-adapter/README.md +0 -136
- package/wab-mcp-adapter/index.js +0 -555
- package/wab-mcp-adapter/package.json +0 -17
package/server/routes/license.js
CHANGED
|
@@ -1,240 +1,240 @@
|
|
|
1
|
-
const express = require('express');
|
|
2
|
-
const crypto = require('crypto');
|
|
3
|
-
const router = express.Router();
|
|
4
|
-
const {
|
|
5
|
-
verifyLicense,
|
|
6
|
-
recordAnalytic,
|
|
7
|
-
findSiteByLicense,
|
|
8
|
-
findSiteById,
|
|
9
|
-
db
|
|
10
|
-
} = require('../models/db');
|
|
11
|
-
const { broadcastAnalytic } = require('../ws');
|
|
12
|
-
const { cache, AnalyticsQueue } = require('../utils/cache');
|
|
13
|
-
const { licenseTokenLimiter, licenseTrackLimiter } = require('../middleware/rateLimits');
|
|
14
|
-
|
|
15
|
-
const analyticsQueue = new AnalyticsQueue(db, { maxSize: 50, maxBufferTotal: 5000 });
|
|
16
|
-
|
|
17
|
-
// ─── Session Token Store (in-memory, TTL 1 hour) ────────────────────
|
|
18
|
-
const sessionTokens = new Map();
|
|
19
|
-
const SESSION_TTL = 60 * 60 * 1000;
|
|
20
|
-
|
|
21
|
-
setInterval(() => {
|
|
22
|
-
const now = Date.now();
|
|
23
|
-
for (const [token, data] of sessionTokens) {
|
|
24
|
-
if (now > data.expiresAt) sessionTokens.delete(token);
|
|
25
|
-
}
|
|
26
|
-
}, 5 * 60 * 1000);
|
|
27
|
-
|
|
28
|
-
function normalizeHost(host) {
|
|
29
|
-
if (!host) return '';
|
|
30
|
-
let h = String(host).toLowerCase().trim();
|
|
31
|
-
if (h.startsWith('www.')) h = h.slice(4);
|
|
32
|
-
return h;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
function getRequestHostname(req) {
|
|
36
|
-
const origin = req.get('origin') || req.get('referer');
|
|
37
|
-
try {
|
|
38
|
-
return origin ? new URL(origin).hostname : req.hostname;
|
|
39
|
-
} catch {
|
|
40
|
-
return req.hostname;
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
function allowDevInsecureOrigin(hostname) {
|
|
45
|
-
if (process.env.NODE_ENV === 'production') return false;
|
|
46
|
-
if (process.env.ALLOW_INSECURE_LICENSE_ORIGIN !== 'true') return false;
|
|
47
|
-
const n = normalizeHost(hostname);
|
|
48
|
-
return n === 'localhost' || n === '127.0.0.1' || n === '[::1]';
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
// ─── Verify (domain + license OR session + siteId) ─────────────────
|
|
52
|
-
router.post('/verify', (req, res) => {
|
|
53
|
-
const { domain, licenseKey, siteId, sessionToken } = req.body;
|
|
54
|
-
|
|
55
|
-
if (sessionToken && siteId) {
|
|
56
|
-
const session = sessionTokens.get(sessionToken);
|
|
57
|
-
if (!session || Date.now() > session.expiresAt) {
|
|
58
|
-
sessionTokens.delete(sessionToken);
|
|
59
|
-
return res.json({ valid: false, error: 'Session expired or invalid', tier: 'free' });
|
|
60
|
-
}
|
|
61
|
-
const requestDomain = getRequestHostname(req);
|
|
62
|
-
if (normalizeHost(requestDomain) !== normalizeHost(session.domain)) {
|
|
63
|
-
return res.json({ valid: false, error: 'Domain mismatch', tier: 'free' });
|
|
64
|
-
}
|
|
65
|
-
if (session.siteId !== siteId) {
|
|
66
|
-
return res.json({ valid: false, error: 'Invalid site', tier: 'free' });
|
|
67
|
-
}
|
|
68
|
-
return res.json({
|
|
69
|
-
valid: true,
|
|
70
|
-
tier: session.tier,
|
|
71
|
-
domain: session.domain,
|
|
72
|
-
allowedPermissions: session.permissions
|
|
73
|
-
});
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
if (!domain || !licenseKey) {
|
|
77
|
-
return res.status(400).json({ valid: false, error: 'Domain and licenseKey are required (or sessionToken + siteId)', tier: 'free' });
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
const cacheKey = `license:${domain}:${licenseKey}`;
|
|
81
|
-
const cached = cache.get(cacheKey);
|
|
82
|
-
if (cached) return res.json(cached);
|
|
83
|
-
|
|
84
|
-
const result = verifyLicense(domain, licenseKey);
|
|
85
|
-
cache.set(cacheKey, result, 60000);
|
|
86
|
-
res.json(result);
|
|
87
|
-
});
|
|
88
|
-
|
|
89
|
-
// ─── Token exchange: siteId (preferred) or licenseKey (legacy) ─────
|
|
90
|
-
router.post('/token', licenseTokenLimiter, (req, res) => {
|
|
91
|
-
const { licenseKey, siteId } = req.body;
|
|
92
|
-
const domain = getRequestHostname(req);
|
|
93
|
-
const normReq = normalizeHost(domain);
|
|
94
|
-
|
|
95
|
-
const finishSession = (site, result) => {
|
|
96
|
-
const sessionToken = crypto.randomBytes(32).toString('hex');
|
|
97
|
-
const expiresAt = Date.now() + SESSION_TTL;
|
|
98
|
-
sessionTokens.set(sessionToken, {
|
|
99
|
-
siteId: site.id,
|
|
100
|
-
domain: site.domain,
|
|
101
|
-
tier: result.tier,
|
|
102
|
-
permissions: result.allowedPermissions,
|
|
103
|
-
expiresAt
|
|
104
|
-
});
|
|
105
|
-
res.json({
|
|
106
|
-
sessionToken,
|
|
107
|
-
siteId: site.id,
|
|
108
|
-
tier: result.tier,
|
|
109
|
-
permissions: result.allowedPermissions,
|
|
110
|
-
expiresIn: SESSION_TTL / 1000
|
|
111
|
-
});
|
|
112
|
-
};
|
|
113
|
-
|
|
114
|
-
if (siteId && !licenseKey) {
|
|
115
|
-
const site = findSiteById.get(siteId);
|
|
116
|
-
if (!site || !site.active) {
|
|
117
|
-
return res.status(404).json({ error: 'Site not found' });
|
|
118
|
-
}
|
|
119
|
-
const originOk =
|
|
120
|
-
normReq === normalizeHost(site.domain) ||
|
|
121
|
-
allowDevInsecureOrigin(domain);
|
|
122
|
-
if (!originOk) {
|
|
123
|
-
return res.status(403).json({ error: 'Origin does not match registered site domain' });
|
|
124
|
-
}
|
|
125
|
-
const cacheKey = `license:${site.domain}:${site.license_key}`;
|
|
126
|
-
let result = cache.get(cacheKey);
|
|
127
|
-
if (!result) {
|
|
128
|
-
result = verifyLicense(site.domain, site.license_key);
|
|
129
|
-
cache.set(cacheKey, result, 60000);
|
|
130
|
-
}
|
|
131
|
-
if (!result.valid) {
|
|
132
|
-
return res.status(403).json({ error: result.error || 'Invalid license', tier: 'free' });
|
|
133
|
-
}
|
|
134
|
-
return finishSession(site, result);
|
|
135
|
-
}
|
|
136
|
-
|
|
137
|
-
if (!licenseKey) {
|
|
138
|
-
return res.status(400).json({ error: 'siteId or licenseKey is required' });
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
const cacheKey = `license:${domain}:${licenseKey}`;
|
|
142
|
-
let result = cache.get(cacheKey);
|
|
143
|
-
if (!result) {
|
|
144
|
-
result = verifyLicense(domain, licenseKey);
|
|
145
|
-
cache.set(cacheKey, result, 60000);
|
|
146
|
-
}
|
|
147
|
-
if (!result.valid) {
|
|
148
|
-
return res.status(403).json({ error: result.error || 'Invalid license', tier: 'free' });
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
const site = findSiteByLicense.get(licenseKey);
|
|
152
|
-
if (!site) {
|
|
153
|
-
return res.status(404).json({ error: 'Site not found' });
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
finishSession(site, result);
|
|
157
|
-
});
|
|
158
|
-
|
|
159
|
-
// ─── Validate Session Token ─────────────────────────────────────────
|
|
160
|
-
router.post('/session', (req, res) => {
|
|
161
|
-
const { sessionToken } = req.body;
|
|
162
|
-
if (!sessionToken) {
|
|
163
|
-
return res.status(400).json({ valid: false, error: 'sessionToken required' });
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
const session = sessionTokens.get(sessionToken);
|
|
167
|
-
if (!session || Date.now() > session.expiresAt) {
|
|
168
|
-
sessionTokens.delete(sessionToken);
|
|
169
|
-
return res.status(401).json({ valid: false, error: 'Session expired or invalid' });
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
const requestDomain = getRequestHostname(req);
|
|
173
|
-
if (normalizeHost(requestDomain) !== normalizeHost(session.domain)) {
|
|
174
|
-
return res.status(403).json({ valid: false, error: 'Domain mismatch' });
|
|
175
|
-
}
|
|
176
|
-
|
|
177
|
-
res.json({
|
|
178
|
-
valid: true,
|
|
179
|
-
siteId: session.siteId,
|
|
180
|
-
tier: session.tier,
|
|
181
|
-
permissions: session.permissions
|
|
182
|
-
});
|
|
183
|
-
});
|
|
184
|
-
|
|
185
|
-
// ─── Analytics track (session-bound; licenseKey deprecated) ──────
|
|
186
|
-
router.post('/track', licenseTrackLimiter, (req, res) => {
|
|
187
|
-
const { sessionToken, actionName, agentId, triggerType, success, metadata, licenseKey } = req.body;
|
|
188
|
-
|
|
189
|
-
if (!actionName) {
|
|
190
|
-
return res.status(400).json({ error: 'actionName is required' });
|
|
191
|
-
}
|
|
192
|
-
|
|
193
|
-
let site;
|
|
194
|
-
if (sessionToken) {
|
|
195
|
-
const session = sessionTokens.get(sessionToken);
|
|
196
|
-
if (!session || Date.now() > session.expiresAt) {
|
|
197
|
-
sessionTokens.delete(sessionToken);
|
|
198
|
-
return res.status(401).json({ error: 'Session expired or invalid' });
|
|
199
|
-
}
|
|
200
|
-
const requestDomain = getRequestHostname(req);
|
|
201
|
-
if (normalizeHost(requestDomain) !== normalizeHost(session.domain)) {
|
|
202
|
-
return res.status(403).json({ error: 'Origin does not match session domain' });
|
|
203
|
-
}
|
|
204
|
-
site = findSiteById.get(session.siteId);
|
|
205
|
-
if (!site || !site.active) {
|
|
206
|
-
return res.status(404).json({ error: 'Site not found' });
|
|
207
|
-
}
|
|
208
|
-
} else if (licenseKey && process.env.ALLOW_LEGACY_LICENSE_TRACK === 'true') {
|
|
209
|
-
site = findSiteByLicense.get(licenseKey);
|
|
210
|
-
if (!site) return res.status(404).json({ error: 'Site not found' });
|
|
211
|
-
} else {
|
|
212
|
-
return res.status(400).json({
|
|
213
|
-
error: 'sessionToken is required. Obtain via POST /api/license/token (see installation snippet).'
|
|
214
|
-
});
|
|
215
|
-
}
|
|
216
|
-
|
|
217
|
-
try {
|
|
218
|
-
analyticsQueue.push({
|
|
219
|
-
siteId: site.id,
|
|
220
|
-
actionName,
|
|
221
|
-
agentId,
|
|
222
|
-
triggerType,
|
|
223
|
-
success: success !== false,
|
|
224
|
-
metadata
|
|
225
|
-
});
|
|
226
|
-
|
|
227
|
-
broadcastAnalytic(site.id, {
|
|
228
|
-
actionName,
|
|
229
|
-
agentId,
|
|
230
|
-
triggerType,
|
|
231
|
-
success: success !== false
|
|
232
|
-
});
|
|
233
|
-
|
|
234
|
-
res.json({ recorded: true });
|
|
235
|
-
} catch (err) {
|
|
236
|
-
res.status(500).json({ error: 'Failed to record analytics' });
|
|
237
|
-
}
|
|
238
|
-
});
|
|
239
|
-
|
|
240
|
-
module.exports = router;
|
|
1
|
+
const express = require('express');
|
|
2
|
+
const crypto = require('crypto');
|
|
3
|
+
const router = express.Router();
|
|
4
|
+
const {
|
|
5
|
+
verifyLicense,
|
|
6
|
+
recordAnalytic,
|
|
7
|
+
findSiteByLicense,
|
|
8
|
+
findSiteById,
|
|
9
|
+
db
|
|
10
|
+
} = require('../models/db');
|
|
11
|
+
const { broadcastAnalytic } = require('../ws');
|
|
12
|
+
const { cache, AnalyticsQueue } = require('../utils/cache');
|
|
13
|
+
const { licenseTokenLimiter, licenseTrackLimiter } = require('../middleware/rateLimits');
|
|
14
|
+
|
|
15
|
+
const analyticsQueue = new AnalyticsQueue(db, { maxSize: 50, maxBufferTotal: 5000 });
|
|
16
|
+
|
|
17
|
+
// ─── Session Token Store (in-memory, TTL 1 hour) ────────────────────
|
|
18
|
+
const sessionTokens = new Map();
|
|
19
|
+
const SESSION_TTL = 60 * 60 * 1000;
|
|
20
|
+
|
|
21
|
+
setInterval(() => {
|
|
22
|
+
const now = Date.now();
|
|
23
|
+
for (const [token, data] of sessionTokens) {
|
|
24
|
+
if (now > data.expiresAt) sessionTokens.delete(token);
|
|
25
|
+
}
|
|
26
|
+
}, 5 * 60 * 1000);
|
|
27
|
+
|
|
28
|
+
function normalizeHost(host) {
|
|
29
|
+
if (!host) return '';
|
|
30
|
+
let h = String(host).toLowerCase().trim();
|
|
31
|
+
if (h.startsWith('www.')) h = h.slice(4);
|
|
32
|
+
return h;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
function getRequestHostname(req) {
|
|
36
|
+
const origin = req.get('origin') || req.get('referer');
|
|
37
|
+
try {
|
|
38
|
+
return origin ? new URL(origin).hostname : req.hostname;
|
|
39
|
+
} catch {
|
|
40
|
+
return req.hostname;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function allowDevInsecureOrigin(hostname) {
|
|
45
|
+
if (process.env.NODE_ENV === 'production') return false;
|
|
46
|
+
if (process.env.ALLOW_INSECURE_LICENSE_ORIGIN !== 'true') return false;
|
|
47
|
+
const n = normalizeHost(hostname);
|
|
48
|
+
return n === 'localhost' || n === '127.0.0.1' || n === '[::1]';
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// ─── Verify (domain + license OR session + siteId) ─────────────────
|
|
52
|
+
router.post('/verify', (req, res) => {
|
|
53
|
+
const { domain, licenseKey, siteId, sessionToken } = req.body;
|
|
54
|
+
|
|
55
|
+
if (sessionToken && siteId) {
|
|
56
|
+
const session = sessionTokens.get(sessionToken);
|
|
57
|
+
if (!session || Date.now() > session.expiresAt) {
|
|
58
|
+
sessionTokens.delete(sessionToken);
|
|
59
|
+
return res.json({ valid: false, error: 'Session expired or invalid', tier: 'free' });
|
|
60
|
+
}
|
|
61
|
+
const requestDomain = getRequestHostname(req);
|
|
62
|
+
if (normalizeHost(requestDomain) !== normalizeHost(session.domain)) {
|
|
63
|
+
return res.json({ valid: false, error: 'Domain mismatch', tier: 'free' });
|
|
64
|
+
}
|
|
65
|
+
if (session.siteId !== siteId) {
|
|
66
|
+
return res.json({ valid: false, error: 'Invalid site', tier: 'free' });
|
|
67
|
+
}
|
|
68
|
+
return res.json({
|
|
69
|
+
valid: true,
|
|
70
|
+
tier: session.tier,
|
|
71
|
+
domain: session.domain,
|
|
72
|
+
allowedPermissions: session.permissions
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
if (!domain || !licenseKey) {
|
|
77
|
+
return res.status(400).json({ valid: false, error: 'Domain and licenseKey are required (or sessionToken + siteId)', tier: 'free' });
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
const cacheKey = `license:${domain}:${licenseKey}`;
|
|
81
|
+
const cached = cache.get(cacheKey);
|
|
82
|
+
if (cached) return res.json(cached);
|
|
83
|
+
|
|
84
|
+
const result = verifyLicense(domain, licenseKey);
|
|
85
|
+
cache.set(cacheKey, result, 60000);
|
|
86
|
+
res.json(result);
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
// ─── Token exchange: siteId (preferred) or licenseKey (legacy) ─────
|
|
90
|
+
router.post('/token', licenseTokenLimiter, (req, res) => {
|
|
91
|
+
const { licenseKey, siteId } = req.body;
|
|
92
|
+
const domain = getRequestHostname(req);
|
|
93
|
+
const normReq = normalizeHost(domain);
|
|
94
|
+
|
|
95
|
+
const finishSession = (site, result) => {
|
|
96
|
+
const sessionToken = crypto.randomBytes(32).toString('hex');
|
|
97
|
+
const expiresAt = Date.now() + SESSION_TTL;
|
|
98
|
+
sessionTokens.set(sessionToken, {
|
|
99
|
+
siteId: site.id,
|
|
100
|
+
domain: site.domain,
|
|
101
|
+
tier: result.tier,
|
|
102
|
+
permissions: result.allowedPermissions,
|
|
103
|
+
expiresAt
|
|
104
|
+
});
|
|
105
|
+
res.json({
|
|
106
|
+
sessionToken,
|
|
107
|
+
siteId: site.id,
|
|
108
|
+
tier: result.tier,
|
|
109
|
+
permissions: result.allowedPermissions,
|
|
110
|
+
expiresIn: SESSION_TTL / 1000
|
|
111
|
+
});
|
|
112
|
+
};
|
|
113
|
+
|
|
114
|
+
if (siteId && !licenseKey) {
|
|
115
|
+
const site = findSiteById.get(siteId);
|
|
116
|
+
if (!site || !site.active) {
|
|
117
|
+
return res.status(404).json({ error: 'Site not found' });
|
|
118
|
+
}
|
|
119
|
+
const originOk =
|
|
120
|
+
normReq === normalizeHost(site.domain) ||
|
|
121
|
+
allowDevInsecureOrigin(domain);
|
|
122
|
+
if (!originOk) {
|
|
123
|
+
return res.status(403).json({ error: 'Origin does not match registered site domain' });
|
|
124
|
+
}
|
|
125
|
+
const cacheKey = `license:${site.domain}:${site.license_key}`;
|
|
126
|
+
let result = cache.get(cacheKey);
|
|
127
|
+
if (!result) {
|
|
128
|
+
result = verifyLicense(site.domain, site.license_key);
|
|
129
|
+
cache.set(cacheKey, result, 60000);
|
|
130
|
+
}
|
|
131
|
+
if (!result.valid) {
|
|
132
|
+
return res.status(403).json({ error: result.error || 'Invalid license', tier: 'free' });
|
|
133
|
+
}
|
|
134
|
+
return finishSession(site, result);
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
if (!licenseKey) {
|
|
138
|
+
return res.status(400).json({ error: 'siteId or licenseKey is required' });
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
const cacheKey = `license:${domain}:${licenseKey}`;
|
|
142
|
+
let result = cache.get(cacheKey);
|
|
143
|
+
if (!result) {
|
|
144
|
+
result = verifyLicense(domain, licenseKey);
|
|
145
|
+
cache.set(cacheKey, result, 60000);
|
|
146
|
+
}
|
|
147
|
+
if (!result.valid) {
|
|
148
|
+
return res.status(403).json({ error: result.error || 'Invalid license', tier: 'free' });
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
const site = findSiteByLicense.get(licenseKey);
|
|
152
|
+
if (!site) {
|
|
153
|
+
return res.status(404).json({ error: 'Site not found' });
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
finishSession(site, result);
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
// ─── Validate Session Token ─────────────────────────────────────────
|
|
160
|
+
router.post('/session', (req, res) => {
|
|
161
|
+
const { sessionToken } = req.body;
|
|
162
|
+
if (!sessionToken) {
|
|
163
|
+
return res.status(400).json({ valid: false, error: 'sessionToken required' });
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
const session = sessionTokens.get(sessionToken);
|
|
167
|
+
if (!session || Date.now() > session.expiresAt) {
|
|
168
|
+
sessionTokens.delete(sessionToken);
|
|
169
|
+
return res.status(401).json({ valid: false, error: 'Session expired or invalid' });
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
const requestDomain = getRequestHostname(req);
|
|
173
|
+
if (normalizeHost(requestDomain) !== normalizeHost(session.domain)) {
|
|
174
|
+
return res.status(403).json({ valid: false, error: 'Domain mismatch' });
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
res.json({
|
|
178
|
+
valid: true,
|
|
179
|
+
siteId: session.siteId,
|
|
180
|
+
tier: session.tier,
|
|
181
|
+
permissions: session.permissions
|
|
182
|
+
});
|
|
183
|
+
});
|
|
184
|
+
|
|
185
|
+
// ─── Analytics track (session-bound; licenseKey deprecated) ──────
|
|
186
|
+
router.post('/track', licenseTrackLimiter, (req, res) => {
|
|
187
|
+
const { sessionToken, actionName, agentId, triggerType, success, metadata, licenseKey } = req.body;
|
|
188
|
+
|
|
189
|
+
if (!actionName) {
|
|
190
|
+
return res.status(400).json({ error: 'actionName is required' });
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
let site;
|
|
194
|
+
if (sessionToken) {
|
|
195
|
+
const session = sessionTokens.get(sessionToken);
|
|
196
|
+
if (!session || Date.now() > session.expiresAt) {
|
|
197
|
+
sessionTokens.delete(sessionToken);
|
|
198
|
+
return res.status(401).json({ error: 'Session expired or invalid' });
|
|
199
|
+
}
|
|
200
|
+
const requestDomain = getRequestHostname(req);
|
|
201
|
+
if (normalizeHost(requestDomain) !== normalizeHost(session.domain)) {
|
|
202
|
+
return res.status(403).json({ error: 'Origin does not match session domain' });
|
|
203
|
+
}
|
|
204
|
+
site = findSiteById.get(session.siteId);
|
|
205
|
+
if (!site || !site.active) {
|
|
206
|
+
return res.status(404).json({ error: 'Site not found' });
|
|
207
|
+
}
|
|
208
|
+
} else if (licenseKey && process.env.ALLOW_LEGACY_LICENSE_TRACK === 'true') {
|
|
209
|
+
site = findSiteByLicense.get(licenseKey);
|
|
210
|
+
if (!site) return res.status(404).json({ error: 'Site not found' });
|
|
211
|
+
} else {
|
|
212
|
+
return res.status(400).json({
|
|
213
|
+
error: 'sessionToken is required. Obtain via POST /api/license/token (see installation snippet).'
|
|
214
|
+
});
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
try {
|
|
218
|
+
analyticsQueue.push({
|
|
219
|
+
siteId: site.id,
|
|
220
|
+
actionName,
|
|
221
|
+
agentId,
|
|
222
|
+
triggerType,
|
|
223
|
+
success: success !== false,
|
|
224
|
+
metadata
|
|
225
|
+
});
|
|
226
|
+
|
|
227
|
+
broadcastAnalytic(site.id, {
|
|
228
|
+
actionName,
|
|
229
|
+
agentId,
|
|
230
|
+
triggerType,
|
|
231
|
+
success: success !== false
|
|
232
|
+
});
|
|
233
|
+
|
|
234
|
+
res.json({ recorded: true });
|
|
235
|
+
} catch (err) {
|
|
236
|
+
res.status(500).json({ error: 'Failed to record analytics' });
|
|
237
|
+
}
|
|
238
|
+
});
|
|
239
|
+
|
|
240
|
+
module.exports = router;
|