暫無描述
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

server.js 33KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637
  1. // Remote Access Platform — backend server
  2. // HTTP JSON API (auth, MFA, machines, audit) + authenticated WebSocket signaling.
  3. const http = require('http');
  4. const https = require('https');
  5. const fs = require('fs');
  6. const path = require('path');
  7. const { WebSocketServer } = require('ws');
  8. const db = require('./db');
  9. const A = require('./auth');
  10. const PORT = process.env.PORT || 8090;
  11. const HTTPS_PORT = process.env.HTTPS_PORT || 8443;
  12. const PUBLIC_DIR = path.join(__dirname, 'public');
  13. const REC_DIR = path.join(__dirname, 'recordings');
  14. try { fs.mkdirSync(REC_DIR, { recursive: true }); } catch (e) {}
  15. const TRANS_DIR = path.join(__dirname, 'transcripts');
  16. try { fs.mkdirSync(TRANS_DIR, { recursive: true }); } catch (e) {}
  17. const SESSION_TTL = 1000 * 60 * 60 * 24; // 24h auto-logout
  18. // ---------- helpers ----------
  19. const now = () => Date.now();
  20. const json = (res, code, body) => {
  21. res.writeHead(code, { 'Content-Type': 'application/json' });
  22. res.end(JSON.stringify(body));
  23. };
  24. function readBody(req) {
  25. return new Promise((resolve) => {
  26. let data = '';
  27. req.on('data', (c) => (data += c));
  28. req.on('end', () => {
  29. try { resolve(data ? JSON.parse(data) : {}); } catch { resolve({}); }
  30. });
  31. });
  32. }
  33. function parseCookies(req) {
  34. const out = {};
  35. (req.headers.cookie || '').split(';').forEach((c) => {
  36. const [k, ...v] = c.trim().split('=');
  37. if (k) out[k] = decodeURIComponent(v.join('='));
  38. });
  39. return out;
  40. }
  41. function audit(entry) {
  42. db.prepare(
  43. `INSERT INTO audit_log (team_id,user_id,user_email,machine_id,machine_name,action,detail,at)
  44. VALUES (@team_id,@user_id,@user_email,@machine_id,@machine_name,@action,@detail,@at)`
  45. ).run({
  46. team_id: entry.team_id, user_id: entry.user_id || null, user_email: entry.user_email || null,
  47. machine_id: entry.machine_id || null, machine_name: entry.machine_name || null,
  48. action: entry.action, detail: entry.detail || null, at: now(),
  49. });
  50. }
  51. // Resolve the logged-in user from the session cookie. Returns user row (with mfa state) or null.
  52. function currentUser(req, { requireMfa = true } = {}) {
  53. const tok = parseCookies(req).sid;
  54. if (!tok) return null;
  55. const s = db.prepare('SELECT * FROM sessions_auth WHERE token=?').get(tok);
  56. if (!s || s.expires_at < now()) return null;
  57. if (requireMfa && !s.mfa_passed) return null;
  58. const u = db.prepare('SELECT * FROM users WHERE id=?').get(s.user_id);
  59. if (!u || u.active === 0) return null;
  60. return { ...u, _session: s };
  61. }
  62. // ---------- HTTP API ----------
  63. const routes = {};
  64. const route = (method, p, fn) => (routes[`${method} ${p}`] = fn);
  65. // Register: creates a team + admin user. MFA must be set up before full access.
  66. route('POST', '/api/register', async (req, res) => {
  67. const anyUser = db.prepare('SELECT 1 FROM users LIMIT 1').get();
  68. if (anyUser && process.env.ALLOW_REGISTRATION !== '1')
  69. return json(res, 403, { error: 'Registration is closed. Contact your administrator.' });
  70. const { email, password, teamName } = await readBody(req);
  71. if (!email || !password) return json(res, 400, { error: 'email and password required' });
  72. if (db.prepare('SELECT 1 FROM users WHERE email=? COLLATE NOCASE').get(email))
  73. return json(res, 409, { error: 'email already registered' });
  74. const teamId = A.id(), userId = A.id();
  75. const { hash, salt } = A.hashPassword(password);
  76. const mfaSecret = A.newMfaSecret();
  77. db.prepare('INSERT INTO teams (id,name,created_at) VALUES (?,?,?)')
  78. .run(teamId, teamName || `${email}'s team`, now());
  79. db.prepare(`INSERT INTO users (id,team_id,email,pw_hash,pw_salt,role,mfa_secret,mfa_enabled,created_at)
  80. VALUES (?,?,?,?,?,?,?,0,?)`)
  81. .run(userId, teamId, email, hash, salt, 'admin', mfaSecret, now());
  82. audit({ team_id: teamId, user_id: userId, user_email: email, action: 'user_registered' });
  83. json(res, 200, { ok: true });
  84. });
  85. // Verify MFA enrollment (confirm the user scanned the QR / entered code)
  86. route('POST', '/api/mfa/enable', async (req, res) => {
  87. const { email, code } = await readBody(req);
  88. const u = db.prepare('SELECT * FROM users WHERE email=? COLLATE NOCASE').get(email);
  89. if (!u) return json(res, 404, { error: 'no such user' });
  90. if (!A.verifyTotp(u.mfa_secret, code)) return json(res, 401, { error: 'invalid code' });
  91. db.prepare('UPDATE users SET mfa_enabled=1 WHERE id=?').run(u.id);
  92. json(res, 200, { ok: true });
  93. });
  94. // Login step 1: email + password -> sets a session cookie (mfa not yet passed)
  95. route('POST', '/api/login', async (req, res) => {
  96. const { email, password, remember } = await readBody(req);
  97. const u = db.prepare('SELECT * FROM users WHERE email=? COLLATE NOCASE').get(email);
  98. if (!u || !A.verifyPassword(password, u.pw_salt, u.pw_hash))
  99. return json(res, 401, { error: 'invalid credentials' });
  100. if (u.active === 0) return json(res, 403, { error: 'This account has been deactivated' });
  101. const tok = A.token();
  102. const ttl = remember ? 1000 * 60 * 60 * 24 * 30 : SESSION_TTL; // 30 days if remembered, else 24h
  103. db.prepare('INSERT INTO sessions_auth (token,user_id,mfa_passed,created_at,expires_at) VALUES (?,?,1,?,?)')
  104. .run(tok, u.id, now(), now() + ttl);
  105. res.setHeader('Set-Cookie', `sid=${tok}; HttpOnly; Path=/; Max-Age=${ttl / 1000}`);
  106. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'login' });
  107. json(res, 200, { ok: true, mfaRequired: false });
  108. });
  109. // Login step 2: TOTP code -> marks session mfa_passed
  110. route('POST', '/api/login/mfa', async (req, res) => {
  111. const { code } = await readBody(req);
  112. const tok = parseCookies(req).sid;
  113. const s = tok && db.prepare('SELECT * FROM sessions_auth WHERE token=?').get(tok);
  114. if (!s) return json(res, 401, { error: 'no session' });
  115. const u = db.prepare('SELECT * FROM users WHERE id=?').get(s.user_id);
  116. if (!A.verifyTotp(u.mfa_secret, code)) return json(res, 401, { error: 'invalid code' });
  117. db.prepare('UPDATE sessions_auth SET mfa_passed=1 WHERE token=?').run(tok);
  118. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'login' });
  119. json(res, 200, { ok: true });
  120. });
  121. route('POST', '/api/logout', async (req, res) => {
  122. const tok = parseCookies(req).sid;
  123. if (tok) db.prepare('DELETE FROM sessions_auth WHERE token=?').run(tok);
  124. res.setHeader('Set-Cookie', 'sid=; HttpOnly; Path=/; Max-Age=0');
  125. json(res, 200, { ok: true });
  126. });
  127. route('GET', '/api/setup-state', async (req, res) => {
  128. const anyUser = db.prepare('SELECT 1 FROM users LIMIT 1').get();
  129. json(res, 200, { registrationOpen: !anyUser || process.env.ALLOW_REGISTRATION === '1' });
  130. });
  131. // ICE servers for WebRTC. Always includes a public STUN; adds managed TURN if
  132. // configured via env (TURN_URLS comma-separated, TURN_USERNAME, TURN_CREDENTIAL).
  133. // Sign up for a managed TURN service (Cloudflare / Metered / Twilio) and set these
  134. // three env vars — nothing to install or run on your side.
  135. route('GET', '/api/ice', async (req, res) => {
  136. const iceServers = [{ urls: 'stun:stun.l.google.com:19302' }];
  137. if (process.env.TURN_URLS) {
  138. iceServers.push({
  139. urls: process.env.TURN_URLS.split(',').map((u) => u.trim()).filter(Boolean),
  140. username: process.env.TURN_USERNAME || '',
  141. credential: process.env.TURN_CREDENTIAL || '',
  142. });
  143. }
  144. json(res, 200, { iceServers });
  145. });
  146. route('GET', '/api/me', async (req, res) => {
  147. const u = currentUser(req);
  148. if (!u) return json(res, 401, { error: 'unauthorized' });
  149. json(res, 200, { id: u.id, email: u.email, role: u.role, teamId: u.team_id, name: u.name || null });
  150. });
  151. // ---------- BizGaze SSO: agent arrives already logged in ----------
  152. route('GET', '/sso', async (req, res) => {
  153. if (!process.env.SSO_SECRET) { res.writeHead(503); return res.end('SSO not configured'); }
  154. const q = new URLSearchParams(req.url.split('?')[1] || '');
  155. const token = q.get('token') || '';
  156. const [payloadB64, sig] = token.split('.');
  157. const fail = (msg) => { res.writeHead(403, { 'Content-Type': 'text/plain' }); res.end(msg); };
  158. if (!payloadB64 || !sig) return fail('Invalid SSO token');
  159. const crypto = require('crypto');
  160. const expect = crypto.createHmac('sha256', process.env.SSO_SECRET).update(payloadB64).digest('base64url');
  161. const sigBuf = Buffer.from(sig), expBuf = Buffer.from(expect);
  162. if (sigBuf.length !== expBuf.length || !crypto.timingSafeEqual(sigBuf, expBuf)) return fail('Invalid SSO signature');
  163. let p; try { p = JSON.parse(Buffer.from(payloadB64, 'base64url').toString()); } catch { return fail('Invalid SSO payload'); }
  164. if (!p.email || !p.exp || p.exp < Math.floor(now() / 1000)) return fail('SSO token expired');
  165. let u = db.prepare('SELECT * FROM users WHERE email=? COLLATE NOCASE').get(p.email);
  166. if (!u) {
  167. const team = db.prepare('SELECT * FROM teams LIMIT 1').get();
  168. if (!team) return fail('No team configured');
  169. const userId = A.id();
  170. const { hash, salt } = A.hashPassword(A.token());
  171. const role = (p.role === 'admin' || p.role === 'viewer') ? p.role : 'technician';
  172. db.prepare(`INSERT INTO users (id,team_id,email,pw_hash,pw_salt,role,name,mfa_secret,mfa_enabled,created_at)
  173. VALUES (?,?,?,?,?,?,?,?,0,?)`)
  174. .run(userId, team.id, p.email, hash, salt, role, (p.name || null), A.newMfaSecret(), now());
  175. u = db.prepare('SELECT * FROM users WHERE id=?').get(userId);
  176. audit({ team_id: team.id, user_id: userId, user_email: p.email, action: 'sso_user_created', detail: p.name || '' });
  177. } else if (p.name && p.name !== u.name) {
  178. db.prepare('UPDATE users SET name=? WHERE id=?').run(p.name, u.id);
  179. }
  180. if (u.active === 0) return fail('Account deactivated');
  181. const tok = A.token();
  182. db.prepare('INSERT INTO sessions_auth (token,user_id,mfa_passed,created_at,expires_at) VALUES (?,?,1,?,?)')
  183. .run(tok, u.id, now(), now() + SESSION_TTL);
  184. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'login', detail: 'via BizGaze SSO' });
  185. const dest = '/connect' + (p.ticket ? ('?ticket=' + encodeURIComponent(p.ticket)) : '');
  186. res.writeHead(302, { 'Set-Cookie': `sid=${tok}; HttpOnly; Path=/; Max-Age=${SESSION_TTL / 1000}`, Location: dest });
  187. res.end();
  188. });
  189. // Admin adds an agent login to their team
  190. route('POST', '/api/users', async (req, res) => {
  191. const u = currentUser(req);
  192. if (!u) return json(res, 401, { error: 'unauthorized' });
  193. if (u.role !== 'admin') return json(res, 403, { error: 'only admins can add agents' });
  194. const { email, password, name, role } = await readBody(req);
  195. if (!email || !password) return json(res, 400, { error: 'email and temporary password required' });
  196. if (db.prepare('SELECT 1 FROM users WHERE email=? COLLATE NOCASE').get(email))
  197. return json(res, 409, { error: 'email already registered' });
  198. const userId = A.id();
  199. const { hash, salt } = A.hashPassword(password);
  200. const mfaSecret = A.newMfaSecret();
  201. const r = (role === 'admin' || role === 'viewer') ? role : 'technician';
  202. db.prepare(`INSERT INTO users (id,team_id,email,pw_hash,pw_salt,role,name,mfa_secret,mfa_enabled,created_at)
  203. VALUES (?,?,?,?,?,?,?,?,0,?)`)
  204. .run(userId, u.team_id, email, hash, salt, r, (name || null), mfaSecret, now());
  205. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'agent_added', detail: email + ' (' + r + ')' });
  206. json(res, 200, { ok: true, id: userId, email, role: r });
  207. });
  208. // List the team's agents
  209. route('GET', '/api/users', async (req, res) => {
  210. const u = currentUser(req);
  211. if (!u) return json(res, 401, { error: 'unauthorized' });
  212. const rows = db.prepare('SELECT id,email,name,role,active,created_at FROM users WHERE team_id=?').all(u.team_id);
  213. json(res, 200, rows);
  214. });
  215. // First-login MFA self-setup: a logged-in (password ok) user who hasn't enabled MFA yet
  216. route('GET', '/api/mfa/setup', async (req, res) => {
  217. const u = currentUser(req, { requireMfa: false });
  218. if (!u) return json(res, 401, { error: 'unauthorized' });
  219. if (u.mfa_enabled) return json(res, 400, { error: 'MFA already enabled' });
  220. json(res, 200, { secret: u.mfa_secret, otpauthUrl: A.otpauthUrl(u.mfa_secret, u.email) });
  221. });
  222. // Admin manages an agent: reset password, rename, deactivate/activate, delete.
  223. // (Display names are owned by the admin/BizGaze app — agents cannot edit them.)
  224. route('POST', '/api/users/manage', async (req, res) => {
  225. const u = currentUser(req);
  226. if (!u) return json(res, 401, { error: 'unauthorized' });
  227. if (u.role !== 'admin') return json(res, 403, { error: 'only admins can manage agents' });
  228. const { id, action, password, name } = await readBody(req);
  229. const target = db.prepare('SELECT * FROM users WHERE id=? AND team_id=?').get(id, u.team_id);
  230. if (!target) return json(res, 404, { error: 'no such agent' });
  231. switch (action) {
  232. case 'reset-password': {
  233. if (!password || String(password).length < 8) return json(res, 400, { error: 'new password must be at least 8 characters' });
  234. const { hash, salt } = A.hashPassword(password);
  235. db.prepare('UPDATE users SET pw_hash=?, pw_salt=? WHERE id=?').run(hash, salt, target.id);
  236. db.prepare('DELETE FROM sessions_auth WHERE user_id=?').run(target.id); // force re-login
  237. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'agent_password_reset', detail: target.email });
  238. return json(res, 200, { ok: true });
  239. }
  240. case 'rename': {
  241. const clean = String(name || '').trim().slice(0, 60);
  242. if (!clean) return json(res, 400, { error: 'name required' });
  243. db.prepare('UPDATE users SET name=? WHERE id=?').run(clean, target.id);
  244. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'agent_renamed', detail: target.email + ' -> ' + clean });
  245. return json(res, 200, { ok: true, name: clean });
  246. }
  247. case 'deactivate': {
  248. if (target.id === u.id) return json(res, 400, { error: 'you cannot deactivate your own account' });
  249. db.prepare('UPDATE users SET active=0 WHERE id=?').run(target.id);
  250. db.prepare('DELETE FROM sessions_auth WHERE user_id=?').run(target.id);
  251. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'agent_deactivated', detail: target.email });
  252. return json(res, 200, { ok: true });
  253. }
  254. case 'activate': {
  255. db.prepare('UPDATE users SET active=1 WHERE id=?').run(target.id);
  256. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'agent_activated', detail: target.email });
  257. return json(res, 200, { ok: true });
  258. }
  259. case 'delete': {
  260. if (target.id === u.id) return json(res, 400, { error: 'you cannot delete your own account' });
  261. db.prepare('DELETE FROM sessions_auth WHERE user_id=?').run(target.id);
  262. db.prepare('DELETE FROM users WHERE id=?').run(target.id);
  263. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'agent_deleted', detail: target.email });
  264. return json(res, 200, { ok: true });
  265. }
  266. default: return json(res, 400, { error: 'unknown action' });
  267. }
  268. });
  269. // Session report: one row per session, filterable by agent and date period
  270. route('GET', '/api/report', async (req, res) => {
  271. const u = currentUser(req);
  272. if (!u) return json(res, 401, { error: 'unauthorized' });
  273. const q = new URLSearchParams(req.url.split('?')[1] || '');
  274. let sql = 'SELECT * FROM sessions_log WHERE team_id=?';
  275. const args = [u.team_id];
  276. if (q.get('agent')) { sql += ' AND agent_email=?'; args.push(q.get('agent')); }
  277. if (q.get('from')) { sql += ' AND started_at>=?'; args.push(new Date(q.get('from') + 'T00:00:00').getTime()); }
  278. if (q.get('to')) { sql += ' AND started_at<=?'; args.push(new Date(q.get('to') + 'T23:59:59').getTime()); }
  279. sql += ' ORDER BY started_at DESC LIMIT 500';
  280. json(res, 200, db.prepare(sql).all(...args));
  281. });
  282. // List machines for the team (with live online status from signaling layer)
  283. route('GET', '/api/machines', async (req, res) => {
  284. const u = currentUser(req);
  285. if (!u) return json(res, 401, { error: 'unauthorized' });
  286. const rows = db.prepare('SELECT id,name,unattended,last_seen FROM machines WHERE team_id=?').all(u.team_id);
  287. json(res, 200, rows.map((m) => ({ ...m, online: onlineAgents.has(m.id) })));
  288. });
  289. // Create a machine enrollment token (admin/technician). Agent uses it to come online.
  290. route('POST', '/api/machines', async (req, res) => {
  291. const u = currentUser(req);
  292. if (!u) return json(res, 401, { error: 'unauthorized' });
  293. if (u.role === 'viewer') return json(res, 403, { error: 'forbidden' });
  294. const { name, unattended } = await readBody(req);
  295. const mId = A.id(), enroll = A.token();
  296. db.prepare('INSERT INTO machines (id,team_id,name,enroll_token,unattended,created_at) VALUES (?,?,?,?,?,?)')
  297. .run(mId, u.team_id, name || 'Unnamed PC', enroll, unattended ? 1 : 0, now());
  298. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, machine_id: mId, machine_name: name, action: 'machine_enrolled' });
  299. json(res, 200, { id: mId, enrollToken: enroll });
  300. });
  301. route('GET', '/api/audit', async (req, res) => {
  302. const u = currentUser(req);
  303. if (!u) return json(res, 401, { error: 'unauthorized' });
  304. const rows = db.prepare("SELECT * FROM audit_log WHERE team_id=? OR team_id='adhoc' ORDER BY at DESC LIMIT 200").all(u.team_id);
  305. json(res, 200, rows);
  306. });
  307. // ---------- session recording: upload (agent) + download (team) ----------
  308. const MAX_REC_BYTES = 500 * 1024 * 1024; // 500 MB safety cap
  309. route('POST', '/api/recording', async (req, res) => {
  310. const u = currentUser(req);
  311. if (!u) return json(res, 401, { error: 'unauthorized' });
  312. const sid = new URLSearchParams(req.url.split('?')[1] || '').get('sessionId');
  313. if (!sid) return json(res, 400, { error: 'sessionId required' });
  314. const row = db.prepare('SELECT * FROM sessions_log WHERE id=? AND team_id=?').get(sid, u.team_id);
  315. if (!row) return json(res, 404, { error: 'no such session' });
  316. const chunks = []; let total = 0, aborted = false;
  317. req.on('data', (c) => { total += c.length; if (total > MAX_REC_BYTES) { aborted = true; req.destroy(); return; } chunks.push(c); });
  318. req.on('end', () => {
  319. if (aborted) return json(res, 413, { error: 'recording too large' });
  320. const fname = sid + '.webm';
  321. try {
  322. fs.writeFileSync(path.join(REC_DIR, fname), Buffer.concat(chunks));
  323. db.prepare('UPDATE sessions_log SET recording=? WHERE id=?').run(fname, sid);
  324. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, action: 'recording_saved', detail: 'session ' + sid });
  325. json(res, 200, { ok: true });
  326. } catch (e) { json(res, 500, { error: 'could not save recording' }); }
  327. });
  328. req.on('error', () => { try { res.end(); } catch (e) {} });
  329. });
  330. route('POST', '/api/transcript', async (req, res) => {
  331. const u = currentUser(req);
  332. if (!u) return json(res, 401, { error: 'unauthorized' });
  333. const sid = new URLSearchParams(req.url.split('?')[1] || '').get('sessionId');
  334. if (!sid) return json(res, 400, { error: 'sessionId required' });
  335. const row = db.prepare('SELECT * FROM sessions_log WHERE id=? AND team_id=?').get(sid, u.team_id);
  336. if (!row) return json(res, 404, { error: 'no such session' });
  337. const chunks = []; let total = 0, aborted = false;
  338. req.on('data', (c) => { total += c.length; if (total > 5 * 1024 * 1024) { aborted = true; req.destroy(); return; } chunks.push(c); });
  339. req.on('end', () => {
  340. if (aborted) return json(res, 413, { error: 'transcript too large' });
  341. const fname = sid + '.txt';
  342. try {
  343. fs.writeFileSync(path.join(TRANS_DIR, fname), Buffer.concat(chunks));
  344. db.prepare('UPDATE sessions_log SET transcript=? WHERE id=?').run(fname, sid);
  345. json(res, 200, { ok: true });
  346. } catch (e) { json(res, 500, { error: 'could not save transcript' }); }
  347. });
  348. req.on('error', () => { try { res.end(); } catch (e) {} });
  349. });
  350. // ---------- static + router ----------
  351. const MIME = { '.html': 'text/html', '.js': 'text/javascript', '.css': 'text/css', '.png': 'image/png', '.jpg': 'image/jpeg', '.jpeg': 'image/jpeg', '.svg': 'image/svg+xml', '.ico': 'image/x-icon' };
  352. function serveStatic(req, res) {
  353. let p = req.url.split('?')[0];
  354. if (p === '/') p = '/index.html';
  355. if (p === '/console') p = '/console.html';
  356. if (p === '/share') p = '/share.html';
  357. if (p === '/connect') p = '/connect.html';
  358. const fp = path.join(PUBLIC_DIR, path.normalize(p));
  359. if (!fp.startsWith(PUBLIC_DIR)) return json(res, 403, { error: 'forbidden' });
  360. fs.readFile(fp, (err, data) => {
  361. if (err) return json(res, 404, { error: 'not found' });
  362. const ct = MIME[path.extname(fp)] || 'application/octet-stream';
  363. res.writeHead(200, { 'Content-Type': ct, 'Cache-Control': 'no-cache' });
  364. res.end(data);
  365. });
  366. }
  367. const server = http.createServer(async (req, res) => {
  368. const key = `${req.method} ${req.url.split('?')[0]}`;
  369. if (routes[key]) return routes[key](req, res);
  370. if (req.method === 'GET' && req.url.split('?')[0].startsWith('/transcripts/')) {
  371. const u = currentUser(req);
  372. if (!u) return json(res, 401, { error: 'unauthorized' });
  373. const name = path.basename(decodeURIComponent(req.url.split('?')[0]));
  374. const sid = name.replace(/\.txt$/i, '');
  375. const row = db.prepare('SELECT * FROM sessions_log WHERE id=? AND team_id=?').get(sid, u.team_id);
  376. if (!row || !row.transcript) return json(res, 404, { error: 'not found' });
  377. const fp = path.join(TRANS_DIR, row.transcript);
  378. if (!fp.startsWith(TRANS_DIR)) return json(res, 403, { error: 'forbidden' });
  379. return fs.stat(fp, (err, st) => {
  380. if (err) return json(res, 404, { error: 'not found' });
  381. res.writeHead(200, { 'Content-Type': 'text/plain; charset=utf-8', 'Content-Length': st.size, 'Content-Disposition': 'attachment; filename="transcript-' + sid + '.txt"', 'Cache-Control': 'no-store' });
  382. const rs = fs.createReadStream(fp);
  383. rs.on('error', () => { try { res.destroy(); } catch (e) {} });
  384. rs.pipe(res);
  385. });
  386. }
  387. if (req.method === 'GET' && req.url.split('?')[0].startsWith('/recordings/')) {
  388. const u = currentUser(req);
  389. if (!u) return json(res, 401, { error: 'unauthorized' });
  390. const name = path.basename(decodeURIComponent(req.url.split('?')[0]));
  391. const sid = name.replace(/\.webm$/i, '');
  392. const row = db.prepare('SELECT * FROM sessions_log WHERE id=? AND team_id=?').get(sid, u.team_id);
  393. if (!row || !row.recording) return json(res, 404, { error: 'not found' });
  394. const fp = path.join(REC_DIR, row.recording);
  395. if (!fp.startsWith(REC_DIR)) return json(res, 403, { error: 'forbidden' });
  396. return fs.stat(fp, (err, st) => {
  397. if (err) return json(res, 404, { error: 'not found' });
  398. res.writeHead(200, { 'Content-Type': 'video/webm', 'Content-Length': st.size, 'Content-Disposition': 'attachment; filename="session-' + sid + '.webm"', 'Cache-Control': 'no-store', 'Accept-Ranges': 'bytes' });
  399. const rs = fs.createReadStream(fp);
  400. rs.on('error', () => { try { res.destroy(); } catch (e) {} });
  401. rs.pipe(res);
  402. });
  403. }
  404. if (req.method === 'GET') return serveStatic(req, res);
  405. json(res, 404, { error: 'not found' });
  406. });
  407. // ---------- WebSocket signaling ----------
  408. // Two kinds of WS clients:
  409. // agent -> authenticates with machine enroll_token, waits for session requests
  410. // viewer -> authenticated technician, requests a session to a machine
  411. // The server brokers consent and relays SDP/ICE. Media never traverses the server.
  412. const onlineAgents = new Map(); // machineId -> { ws, machine }
  413. const liveSessions = new Map(); // sessionId -> { agentWs, viewerWs, machine, user }
  414. const pendingShares = new Map(); // code -> { sharerWs, sessionId } (no-install ad-hoc shares)
  415. function onConnection(ws, req) {
  416. const hb = setInterval(() => {
  417. if (ws.readyState === 1) { try { ws.ping(); } catch {} } else { clearInterval(hb); }
  418. }, 25000);
  419. ws.on('message', (raw) => {
  420. let m; try { m = JSON.parse(raw); } catch { return; }
  421. handle(ws, m, req);
  422. });
  423. ws.on('close', () => { clearInterval(hb); cleanup(ws); });
  424. }
  425. const wss = new WebSocketServer({ server, path: '/ws' });
  426. wss.on('connection', onConnection);
  427. function handle(ws, m, req) {
  428. switch (m.type) {
  429. // --- Agent comes online ---
  430. case 'agent-hello': {
  431. const machine = db.prepare('SELECT * FROM machines WHERE enroll_token=?').get(m.enrollToken);
  432. if (!machine) return ws.send(JSON.stringify({ type: 'error', message: 'invalid enroll token' }));
  433. ws.kind = 'agent'; ws.machineId = machine.id;
  434. onlineAgents.set(machine.id, { ws, machine });
  435. db.prepare('UPDATE machines SET last_seen=? WHERE id=?').run(now(), machine.id);
  436. ws.send(JSON.stringify({ type: 'agent-registered', machineId: machine.id, name: machine.name }));
  437. break;
  438. }
  439. // --- Technician requests control of a machine ---
  440. case 'viewer-connect': {
  441. const u = currentUser(req); // cookie sent on WS upgrade
  442. if (!u) return ws.send(JSON.stringify({ type: 'error', message: 'unauthorized' }));
  443. const agent = onlineAgents.get(m.machineId);
  444. const machine = db.prepare('SELECT * FROM machines WHERE id=? AND team_id=?').get(m.machineId, u.team_id);
  445. if (!machine) return ws.send(JSON.stringify({ type: 'error', message: 'no such machine' }));
  446. if (!agent) return ws.send(JSON.stringify({ type: 'error', message: 'machine offline' }));
  447. if (u.role === 'viewer' && false) {} // view-only still allowed to watch; control gated agent-side
  448. const sessionId = A.token(8);
  449. ws.kind = 'viewer'; ws.sessionId = sessionId;
  450. liveSessions.set(sessionId, { agentWs: agent.ws, viewerWs: ws, machine, user: u });
  451. audit({ team_id: u.team_id, user_id: u.id, user_email: u.email, machine_id: machine.id, machine_name: machine.name, action: 'session_requested' });
  452. // Ask the agent for consent (or auto-grant if unattended policy is on)
  453. agent.ws.sessionId = sessionId;
  454. agent.ws.send(JSON.stringify({
  455. type: 'session-request', sessionId,
  456. technician: u.email, unattended: !!machine.unattended,
  457. }));
  458. ws.send(JSON.stringify({ type: 'session-pending', sessionId, machineName: machine.name }));
  459. break;
  460. }
  461. // --- Agent grants/denies consent ---
  462. case 'consent': {
  463. const sess = liveSessions.get(m.sessionId);
  464. if (!sess) return;
  465. if (m.granted) {
  466. audit({ team_id: sess.machine.team_id, user_id: sess.user.id, user_email: sess.user.email, machine_id: sess.machine.id, machine_name: sess.machine.name, action: 'consent_granted', detail: sess.ticket ? 'Ticket ' + sess.ticket : (sess.machine.id ? null : 'Direct session') });
  467. try {
  468. db.prepare('INSERT INTO sessions_log (id,team_id,agent_email,agent_name,ticket,started_at) VALUES (?,?,?,?,?,?)')
  469. .run(m.sessionId, sess.machine.team_id, sess.user.email, (sess.agentName || sess.user.email), sess.ticket || null, now());
  470. } catch (e) { /* duplicate consent */ }
  471. sess.viewerWs.send(JSON.stringify({ type: 'session-ready', sessionId: m.sessionId }));
  472. sess.agentWs.send(JSON.stringify({ type: 'start-stream', sessionId: m.sessionId }));
  473. } else {
  474. audit({ team_id: sess.machine.team_id, user_id: sess.user.id, user_email: sess.user.email, machine_id: sess.machine.id, machine_name: sess.machine.name, action: 'consent_denied', detail: sess.ticket ? 'Ticket ' + sess.ticket : (sess.machine.id ? null : 'Direct session') });
  475. sess.viewerWs.send(JSON.stringify({ type: 'session-denied', sessionId: m.sessionId }));
  476. liveSessions.delete(m.sessionId);
  477. }
  478. break;
  479. }
  480. // --- No-install: end user opens /share, gets a one-time code ---
  481. case 'share-create': {
  482. let code;
  483. do { code = A.numericCode(6); } while (pendingShares.has(code));
  484. const sessionId = A.token(8);
  485. ws.kind = 'sharer'; ws.shareCode = code; ws.sessionId = sessionId;
  486. pendingShares.set(code, { sharerWs: ws, sessionId });
  487. ws.send(JSON.stringify({ type: 'share-code', code }));
  488. break;
  489. }
  490. // --- Logged-in agent enters the code (+ ticket) to connect ---
  491. case 'code-connect': {
  492. const agent = currentUser(req); // identity from the agent's authenticated session
  493. if (!agent) {
  494. return ws.send(JSON.stringify({ type: 'error', message: 'Please sign in as an agent first' }));
  495. }
  496. const ticket = String(m.ticket || '').trim() || null; // optional: direct sessions have no ticket
  497. const pend = pendingShares.get(String(m.code || '').trim());
  498. if (!pend || pend.sharerWs.readyState !== 1) {
  499. return ws.send(JSON.stringify({ type: 'error', message: 'Invalid or expired code' }));
  500. }
  501. pendingShares.delete(pend.sharerWs.shareCode);
  502. const sessionId = pend.sessionId;
  503. ws.kind = 'viewer'; ws.sessionId = sessionId;
  504. const agentName = agent.name || agent.email;
  505. const machine = { id: null, name: 'Support session ' + pend.sharerWs.shareCode, team_id: agent.team_id };
  506. const user = { id: agent.id, email: agent.email, team_id: agent.team_id };
  507. liveSessions.set(sessionId, { agentWs: pend.sharerWs, viewerWs: ws, machine, user, ticket, agentName });
  508. pend.sharerWs.sessionId = sessionId;
  509. audit({ team_id: agent.team_id, user_id: agent.id, user_email: agent.email, machine_name: machine.name, action: 'code_session_requested', detail: (ticket ? 'Ticket ' + ticket : 'Direct session (no ticket)') + ' · agent ' + agentName });
  510. pend.sharerWs.send(JSON.stringify({ type: 'share-request', sessionId, technician: agentName, ticket }));
  511. ws.send(JSON.stringify({ type: 'code-pending', sessionId }));
  512. break;
  513. }
  514. // --- Relay WebRTC signaling between the two peers ---
  515. case 'offer': case 'answer': case 'ice-candidate': {
  516. const sess = liveSessions.get(m.sessionId || ws.sessionId);
  517. if (!sess) return;
  518. const peer = ws === sess.agentWs ? sess.viewerWs : sess.agentWs;
  519. if (peer && peer.readyState === 1) peer.send(JSON.stringify(m));
  520. break;
  521. }
  522. case 'transcript': {
  523. const sess = liveSessions.get(m.sessionId || ws.sessionId);
  524. if (!sess) return;
  525. const peer = ws === sess.agentWs ? sess.viewerWs : sess.agentWs;
  526. if (peer && peer.readyState === 1) peer.send(JSON.stringify(m));
  527. break;
  528. }
  529. case 'recording': {
  530. const sess = liveSessions.get(m.sessionId || ws.sessionId);
  531. if (!sess) return;
  532. const peer = ws === sess.agentWs ? sess.viewerWs : sess.agentWs;
  533. if (peer && peer.readyState === 1) peer.send(JSON.stringify(m));
  534. break;
  535. }
  536. case 'end-session': {
  537. endSession(ws.sessionId, m.reason || null);
  538. break;
  539. }
  540. }
  541. }
  542. function notifyBizGaze(sessionId) {
  543. const url = process.env.BIZGAZE_WEBHOOK_URL;
  544. if (!url) return;
  545. try {
  546. const row = db.prepare('SELECT * FROM sessions_log WHERE id=?').get(sessionId);
  547. if (!row) return;
  548. const body = JSON.stringify({ event: 'session.ended', sessionId: row.id, agent_email: row.agent_email,
  549. agent_name: row.agent_name, ticket: row.ticket, started_at: row.started_at, ended_at: row.ended_at,
  550. duration_ms: row.ended_at ? row.ended_at - row.started_at : null });
  551. const crypto = require('crypto');
  552. const sig = process.env.SSO_SECRET ? crypto.createHmac('sha256', process.env.SSO_SECRET).update(body).digest('base64url') : '';
  553. fetch(url, { method: 'POST', headers: { 'Content-Type': 'application/json', 'X-BizGaze-Signature': sig }, body }).catch(()=>{});
  554. } catch (e) {}
  555. }
  556. function endSession(sessionId, reason) {
  557. const sess = liveSessions.get(sessionId);
  558. if (!sess) return;
  559. try { db.prepare('UPDATE sessions_log SET ended_at=? WHERE id=? AND ended_at IS NULL').run(now(), sessionId); } catch (e) {}
  560. notifyBizGaze(sessionId);
  561. audit({ team_id: sess.machine.team_id, user_id: sess.user.id, user_email: sess.user.email, machine_id: sess.machine.id, machine_name: sess.machine.name, action: 'session_ended', detail: sess.ticket ? 'Ticket ' + sess.ticket : (sess.machine.id ? null : 'Direct session') });
  562. [sess.agentWs, sess.viewerWs].forEach((p) => {
  563. if (p && p.readyState === 1) p.send(JSON.stringify({ type: 'session-ended', sessionId, reason: reason || null }));
  564. });
  565. liveSessions.delete(sessionId);
  566. }
  567. function cleanup(ws) {
  568. if (ws.kind === 'agent' && ws.machineId) onlineAgents.delete(ws.machineId);
  569. if (ws.kind === 'sharer' && ws.shareCode) pendingShares.delete(ws.shareCode);
  570. if (ws.sessionId) {
  571. for (const [sid, sess] of liveSessions) {
  572. if (sess.agentWs === ws || sess.viewerWs === ws) endSession(sid);
  573. }
  574. }
  575. }
  576. server.listen(PORT, () => {
  577. console.log(`HTTP on http://localhost:${PORT}`);
  578. });
  579. // HTTPS — required so other devices can share their screen (browsers block
  580. // screen capture on non-secure origins). Uses cert.pem/key.pem if present.
  581. let httpsServer = null;
  582. try {
  583. const certPath = path.join(__dirname, 'cert.pem');
  584. const keyPath = path.join(__dirname, 'key.pem');
  585. if (fs.existsSync(certPath) && fs.existsSync(keyPath)) {
  586. httpsServer = https.createServer(
  587. { cert: fs.readFileSync(certPath), key: fs.readFileSync(keyPath) },
  588. (req, res) => server.emit('request', req, res)
  589. );
  590. const wssSecure = new WebSocketServer({ server: httpsServer, path: '/ws' });
  591. wssSecure.on('connection', onConnection);
  592. httpsServer.listen(HTTPS_PORT, () => {
  593. console.log(`HTTPS on https://localhost:${HTTPS_PORT} (use this address from other devices)`);
  594. console.log(` End user shares screen: https://<this-pc-ip>:${HTTPS_PORT}/share`);
  595. console.log(` Technician connects: https://<this-pc-ip>:${HTTPS_PORT}/connect`);
  596. });
  597. } else {
  598. console.log('(No cert.pem/key.pem found — HTTPS disabled. Other devices can view but not share their screen.)');
  599. }
  600. } catch (e) {
  601. console.log('HTTPS failed to start:', e.message);
  602. }
  603. module.exports = { server };