create-volt 0.43.0 → 0.45.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/CHANGELOG.md CHANGED
@@ -4,6 +4,25 @@ All notable changes to `create-volt` are documented here. The format follows
4
4
  [Keep a Changelog](https://keepachangelog.com/), and this project adheres to
5
5
  [Semantic Versioning](https://semver.org/).
6
6
 
7
+ ## [0.45.0] - 2026-06-29
8
+
9
+ ### Added
10
+ - **Buy AI credits from the config wizard.** When an app uses the hosted gateway
11
+ (`VOLT_AI_TOKEN` set), the `--edit` wizard shows an **AI credits** card — live
12
+ balance + tier, and top-up buttons that open Stripe Checkout. The purchase flow
13
+ lives in the (shell-gated) config only; the running app never exposes it.
14
+ Proxied via `/setup/ai-credits` + `/setup/ai-credits/checkout` to the gateway.
15
+
16
+ ## [0.44.0] - 2026-06-29
17
+
18
+ ### Added
19
+ - **`--logs` — a built-in log viewer** on its own localhost port (like `--studio`):
20
+ `npm run logs`. Tails pm2 stdout/stderr **out of the box**; an Analytics tab
21
+ parses Apache/nginx access logs via `mir-sentinel` (optional dep) → top
22
+ paths/status/IPs + bot/attack counts. **Add more sources** (other apps, servers,
23
+ mounted/tunneled paths) right in the viewer — saved to `.volt/logs.json`.
24
+ Localhost-only; SSH-tunnel the port for a remote box.
25
+
7
26
  ## [0.43.0] - 2026-06-30
8
27
 
9
28
  ### Added
@@ -574,6 +593,8 @@ All notable changes to `create-volt` are documented here. The format follows
574
593
  watching and full-page hot reload. Supports `--skip-install` and `--force`,
575
594
  and auto-detects npm / pnpm / yarn / bun for the install step.
576
595
 
596
+ [0.45.0]: https://github.com/MIR-2025/volt/releases/tag/v0.45.0
597
+ [0.44.0]: https://github.com/MIR-2025/volt/releases/tag/v0.44.0
577
598
  [0.43.0]: https://github.com/MIR-2025/volt/releases/tag/v0.43.0
578
599
  [0.42.0]: https://github.com/MIR-2025/volt/releases/tag/v0.42.0
579
600
  [0.41.0]: https://github.com/MIR-2025/volt/releases/tag/v0.41.0
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "create-volt",
3
- "version": "0.43.0",
3
+ "version": "0.45.0",
4
4
  "description": "Scaffold a new Volt app — no-build, signals-based UI with Socket.io hot reload.",
5
5
  "type": "module",
6
6
  "bin": {
@@ -11,7 +11,8 @@
11
11
  "pm2": "npx --yes pm2 start ecosystem.config.cjs",
12
12
  "pm2:restart": "npx --yes pm2 restart ecosystem.config.cjs",
13
13
  "pm2:stop": "npx --yes pm2 delete ecosystem.config.cjs",
14
- "pm2:logs": "npx --yes pm2 logs"
14
+ "pm2:logs": "npx --yes pm2 logs",
15
+ "logs": "node server.js --logs"
15
16
  },
16
17
  "dependencies": {
17
18
  "express": "^4.22.2",
@@ -13,6 +13,7 @@ import fs from "node:fs";
13
13
  import path from "node:path";
14
14
  import { spawn, spawnSync } from "node:child_process";
15
15
  import { fileURLToPath, pathToFileURL } from "node:url";
16
+ import os from "node:os";
16
17
  import express from "express";
17
18
  import { Server as SocketServer } from "socket.io";
18
19
 
@@ -340,6 +341,42 @@ function startSetup() {
340
341
  }
341
342
  return;
342
343
  }
344
+ // --- AI credits: in-config purchase flow. Proxies the hosted gateway with
345
+ // the app's VOLT_AI_TOKEN — the buy flow lives here in the (shell-gated)
346
+ // config only, never in the running app. ---
347
+ if (req.method === "GET" && p === "/setup/ai-credits") {
348
+ const env = readEnvFile();
349
+ res.setHeader("Content-Type", "application/json");
350
+ if (!env.VOLT_AI_TOKEN) return res.end(JSON.stringify({ ok: false, error: "no VOLT_AI_TOKEN (BYO or unset)" }));
351
+ const base = (env.VOLT_AI_GATEWAY || "https://voltjs.com/api/ai").replace(/\/api\/ai\/?$/, "");
352
+ fetch(base + "/api/credits", { headers: { authorization: "Bearer " + env.VOLT_AI_TOKEN } })
353
+ .then((r) => r.json())
354
+ .then((j) => res.end(JSON.stringify(j)))
355
+ .catch(() => res.end(JSON.stringify({ ok: false, error: "gateway unreachable" })));
356
+ return;
357
+ }
358
+ if (req.method === "POST" && p === "/setup/ai-credits/checkout") {
359
+ let cbody = "";
360
+ req.on("data", (c) => (cbody += c));
361
+ req.on("end", () => {
362
+ const env = readEnvFile();
363
+ res.setHeader("Content-Type", "application/json");
364
+ if (!env.VOLT_AI_TOKEN) return res.end(JSON.stringify({ ok: false, error: "no VOLT_AI_TOKEN" }));
365
+ const base = (env.VOLT_AI_GATEWAY || "https://voltjs.com/api/ai").replace(/\/api\/ai\/?$/, "");
366
+ let amountUsd = 0;
367
+ try {
368
+ amountUsd = Number(JSON.parse(cbody || "{}").amountUsd) || 0;
369
+ } catch {
370
+ /* bad json */
371
+ }
372
+ const baseUrl = env.SITE_URL || `http://localhost:${configPort()}`;
373
+ fetch(base + "/api/credits/checkout", { method: "POST", headers: { "content-type": "application/json", authorization: "Bearer " + env.VOLT_AI_TOKEN }, body: JSON.stringify({ amountUsd, baseUrl }) })
374
+ .then((r) => r.json())
375
+ .then((j) => res.end(JSON.stringify(j)))
376
+ .catch(() => res.end(JSON.stringify({ ok: false, error: "gateway unreachable" })));
377
+ });
378
+ return;
379
+ }
343
380
  // --- content manager: list / read / write / delete pages + posts ---
344
381
  if (req.method === "GET" && p === "/setup/content") {
345
382
  const list = (type) => {
@@ -614,7 +651,118 @@ async function startStudio() {
614
651
  });
615
652
  }
616
653
 
617
- // --- gate: studio / setup (first run, --edit) / the app ---
654
+ // --- `--logs`: a disposable, localhost-only log viewer (its own port, like
655
+ // --studio). Tails pm2 stdout/stderr; with mir-sentinel installed, an Analytics
656
+ // tab parses Apache/nginx access logs (ACCESS_LOG). Shell access is the auth;
657
+ // for a remote box, SSH-tunnel the port. ---
658
+ async function startLogs() {
659
+ loadEnv();
660
+ const PORT = configPort();
661
+ const name = (() => {
662
+ try {
663
+ return JSON.parse(fs.readFileSync(PKG_PATH, "utf8")).name || "app";
664
+ } catch {
665
+ return "app";
666
+ }
667
+ })();
668
+ const logsDir = path.join(os.homedir(), ".pm2", "logs");
669
+ // Sources: pm2 stdout/stderr work out of the box; ACCESS_LOG adds an Apache/nginx
670
+ // log; users add more (other apps/servers/mounted or tunneled paths) via
671
+ // .volt/logs.json, editable here in the viewer. Re-read per request so additions
672
+ // show live. For a remote box, ship its log here or SSH-tunnel + run --logs there.
673
+ const LOGS_JSON = path.join(__dirname, ".volt", "logs.json");
674
+ const readExtra = () => {
675
+ try {
676
+ const a = JSON.parse(fs.readFileSync(LOGS_JSON, "utf8"));
677
+ return Array.isArray(a) ? a.filter((x) => x && x.label && x.file) : [];
678
+ } catch {
679
+ return [];
680
+ }
681
+ };
682
+ const sources = () => {
683
+ const s = { app: path.join(logsDir, `${name}-out.log`), error: path.join(logsDir, `${name}-error.log`) };
684
+ if (process.env.ACCESS_LOG) s.access = process.env.ACCESS_LOG;
685
+ for (const x of readExtra()) s[x.label] = x.file;
686
+ return s;
687
+ };
688
+ const tail = (f, n) => (f && fs.existsSync(f) ? fs.readFileSync(f, "utf8").split(/\r?\n/).filter(Boolean).slice(-n) : []);
689
+ let parseLine = null;
690
+ try {
691
+ parseLine = (await import("mir-sentinel")).parseLine;
692
+ } catch {
693
+ /* analytics optional */
694
+ }
695
+ const top = (arr, key) => {
696
+ const m = {};
697
+ for (const x of arr) if (x && x[key]) m[x[key]] = (m[x[key]] || 0) + 1;
698
+ return Object.entries(m).sort((a, b) => b[1] - a[1]).slice(0, 10);
699
+ };
700
+ const assets = {
701
+ "/": ["text/html; charset=utf-8", fs.readFileSync(path.join(__dirname, "setup", "logs.html"))],
702
+ "/logs.js": ["text/javascript; charset=utf-8", fs.readFileSync(path.join(__dirname, "setup", "logs.js"))],
703
+ };
704
+ const json = (res, o) => {
705
+ res.setHeader("Content-Type", "application/json");
706
+ res.end(JSON.stringify(o));
707
+ };
708
+ const server = http.createServer((req, res) => {
709
+ const u = new URL(req.url, "http://localhost");
710
+ const p = u.pathname;
711
+ if (assets[p]) {
712
+ res.setHeader("Content-Type", assets[p][0]);
713
+ return res.end(assets[p][1]);
714
+ }
715
+ if (p === "/api/sources") return json(res, { sources: Object.keys(sources()), extra: readExtra(), analytics: !!parseLine });
716
+ if (p === "/api/tail") {
717
+ const f = sources()[u.searchParams.get("source")];
718
+ return f ? json(res, { ok: true, lines: tail(f, Math.min(2000, Number(u.searchParams.get("lines")) || 300)) }) : json(res, { ok: false });
719
+ }
720
+ if (p === "/api/analytics") {
721
+ if (!parseLine) return json(res, { ok: false, error: "npm i mir-sentinel for analytics" });
722
+ const f = sources()[u.searchParams.get("source")];
723
+ if (!f) return json(res, { ok: false });
724
+ const parsed = tail(f, 5000).map((l) => parseLine(l));
725
+ return json(res, { ok: true, total: parsed.length, paths: top(parsed, "path"), statuses: top(parsed, "status"), ips: top(parsed, "ip"), bots: parsed.filter((x) => x && x.bot).length, attacks: parsed.filter((x) => x && x.attack).length });
726
+ }
727
+ // add/remove a source ("add servers") — written to .volt/logs.json
728
+ if (req.method === "POST" && (p === "/api/source" || p === "/api/source/remove")) {
729
+ let body = "";
730
+ req.on("data", (c) => (body += c));
731
+ req.on("end", () => {
732
+ try {
733
+ const { label, file } = JSON.parse(body || "{}");
734
+ if (!/^[a-z0-9][a-z0-9 _-]*$/i.test(label || "")) throw new Error("label: letters, numbers, spaces, - _");
735
+ let list = readExtra().filter((x) => x.label !== label);
736
+ if (p === "/api/source") list.push({ label, file: String(file || "") });
737
+ fs.mkdirSync(path.dirname(LOGS_JSON), { recursive: true });
738
+ fs.writeFileSync(LOGS_JSON, JSON.stringify(list, null, 2));
739
+ json(res, { ok: true });
740
+ } catch (e) {
741
+ res.statusCode = 400;
742
+ json(res, { ok: false, error: e.message });
743
+ }
744
+ });
745
+ return;
746
+ }
747
+ res.statusCode = 404;
748
+ res.end("not found");
749
+ });
750
+ server.on("error", (e) => {
751
+ if (e.code === "EADDRINUSE") {
752
+ console.error(`\n[volt] Logs port ${PORT} is in use — set CONFIG_PORT in .env or pass --port <n>.`);
753
+ process.exit(1);
754
+ }
755
+ throw e;
756
+ });
757
+ server.listen(PORT, "127.0.0.1", () => {
758
+ const url = `http://localhost:${PORT}`;
759
+ console.log(`\nVolt logs at ${url} (${parseLine ? "analytics on" : "raw tail — npm i mir-sentinel for analytics"})`);
760
+ console.log(" localhost only; for a remote box: ssh -L " + PORT + ":localhost:" + PORT + " you@server");
761
+ openBrowser(url);
762
+ });
763
+ }
764
+
765
+ // --- gate: studio / logs / setup (first run, --edit) / the app ---
618
766
  const editMode = process.argv.includes("--edit") || process.argv.includes("-e");
619
767
  // In production / on a PaaS there's no interactive wizard: config comes from the
620
768
  // platform's env vars (a Dockerfile sets NODE_ENV=production). Only fall back to
@@ -622,6 +770,8 @@ const editMode = process.argv.includes("--edit") || process.argv.includes("-e");
622
770
  const configured = fs.existsSync(ENV_PATH) || process.env.VOLT_ADDONS != null || process.env.NODE_ENV === "production";
623
771
  if (process.argv.includes("--studio")) {
624
772
  startStudio();
773
+ } else if (process.argv.includes("--logs")) {
774
+ startLogs();
625
775
  } else if (editMode || !configured) {
626
776
  startSetup();
627
777
  } else {
@@ -0,0 +1,29 @@
1
+ <!doctype html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="utf-8" /><meta name="viewport" content="width=device-width, initial-scale=1" />
5
+ <title>Logs — Volt</title><meta name="robots" content="noindex" /><link rel="icon" href="/favicon.webp" />
6
+ <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/css/bootstrap.min.css" rel="stylesheet" />
7
+ <style>body{background:#0f1115;color:#e7e9ee}pre{background:#0b0d11;color:#cfe3ff;border:1px solid #232a36;border-radius:8px;padding:12px;max-height:72vh;overflow:auto;font-size:12.5px;white-space:pre-wrap}</style>
8
+ </head>
9
+ <body>
10
+ <div class="container-fluid py-3">
11
+ <div class="d-flex gap-2 align-items-center mb-2 flex-wrap">
12
+ <strong class="me-2">Volt logs</strong>
13
+ <select id="src" class="form-select form-select-sm" style="max-width:200px"></select>
14
+ <select id="view" class="form-select form-select-sm" style="max-width:160px"><option value="tail">Raw tail</option><option value="analytics">Analytics</option></select>
15
+ <button id="refresh" class="btn btn-sm btn-outline-secondary">Refresh</button>
16
+ <button id="toggleadd" class="btn btn-sm btn-outline-secondary" title="Add another log source (server, app, mounted/tunneled path)">+ source</button>
17
+ <label class="form-check-label small ms-1"><input id="follow" type="checkbox" class="form-check-input" /> follow</label>
18
+ <input id="filter" class="form-control form-control-sm ms-auto" style="max-width:240px" placeholder="filter…" />
19
+ </div>
20
+ <div id="addbar" class="d-flex gap-2 mb-2" style="display:none">
21
+ <input id="newlabel" class="form-control form-control-sm" style="max-width:160px" placeholder="label (e.g. prod)" />
22
+ <input id="newfile" class="form-control form-control-sm" style="max-width:420px" placeholder="/var/log/nginx/access.log (local, mounted, or shipped path)" />
23
+ <button id="addsrc" class="btn btn-sm btn-primary">Add</button>
24
+ </div>
25
+ <div id="out"></div>
26
+ </div>
27
+ <script type="module" src="/logs.js"></script>
28
+ </body>
29
+ </html>
@@ -0,0 +1,58 @@
1
+ // logs.js — the --logs viewer client (raw tail + mir-sentinel analytics).
2
+ const $ = (s) => document.querySelector(s);
3
+ const esc = (s) => String(s).replace(/[&<>]/g, (c) => ({ "&": "&amp;", "<": "&lt;", ">": "&gt;" })[c]);
4
+ let timer = null;
5
+
6
+ async function loadSources() {
7
+ const { sources = [] } = await (await fetch("/api/sources")).json();
8
+ $("#src").innerHTML = sources.map((s) => `<option value="${esc(s)}">${esc(s)}</option>`).join("") || `<option>(no logs found)</option>`;
9
+ }
10
+
11
+ async function render() {
12
+ const src = $("#src").value;
13
+ if (!src) return;
14
+ if ($("#view").value === "analytics") {
15
+ const a = await (await fetch(`/api/analytics?source=${encodeURIComponent(src)}`)).json();
16
+ if (!a.ok) {
17
+ $("#out").innerHTML = `<div class="text-muted small">${esc(a.error || "no analytics")}</div>`;
18
+ return;
19
+ }
20
+ const tbl = (title, rows) => `<h6 class="mt-3">${title}</h6><table class="table table-dark table-sm mb-0"><tbody>${(rows || []).map(([k, v]) => `<tr><td>${esc(String(k))}</td><td class="text-end">${v}</td></tr>`).join("")}</tbody></table>`;
21
+ $("#out").innerHTML = `<div class="small text-muted mb-2">${a.total} lines · ${a.bots} bot · ${a.attacks} attack</div>` + tbl("Top paths", a.paths) + tbl("Status codes", a.statuses) + tbl("Top IPs", a.ips);
22
+ } else {
23
+ const { lines = [] } = await (await fetch(`/api/tail?source=${encodeURIComponent(src)}&lines=400`)).json();
24
+ const filter = $("#filter").value.toLowerCase();
25
+ const shown = filter ? lines.filter((l) => l.toLowerCase().includes(filter)) : lines;
26
+ const pre = document.createElement("pre");
27
+ pre.textContent = shown.join("\n") || "(empty)";
28
+ $("#out").innerHTML = "";
29
+ $("#out").appendChild(pre);
30
+ pre.scrollTop = pre.scrollHeight;
31
+ }
32
+ }
33
+
34
+ $("#toggleadd").onclick = () => {
35
+ const b = $("#addbar");
36
+ b.style.display = b.style.display === "none" ? "flex" : "none";
37
+ };
38
+ $("#addsrc").onclick = async () => {
39
+ const label = $("#newlabel").value.trim();
40
+ const file = $("#newfile").value.trim();
41
+ if (!label || !file) return;
42
+ const r = await (await fetch("/api/source", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ label, file }) })).json();
43
+ if (!r.ok) return alert(r.error || "could not add source");
44
+ $("#newlabel").value = $("#newfile").value = "";
45
+ $("#addbar").style.display = "none";
46
+ await loadSources();
47
+ $("#src").value = label;
48
+ render();
49
+ };
50
+ $("#refresh").onclick = render;
51
+ $("#src").onchange = render;
52
+ $("#view").onchange = render;
53
+ $("#filter").oninput = render;
54
+ $("#follow").onchange = (e) => {
55
+ clearInterval(timer);
56
+ if (e.target.checked) timer = setInterval(render, 3000);
57
+ };
58
+ loadSources().then(render);
@@ -284,6 +284,21 @@ async function doUpgrade() {
284
284
  status("Upgrade request failed.");
285
285
  }
286
286
  }
287
+
288
+ // AI credits — config-only purchase flow (gateway mode). Hidden unless a
289
+ // VOLT_AI_TOKEN is set and the gateway answers.
290
+ const aiCredits = signal(null); // { ok, tier, creditBalanceUsd, payments } | { ok:false }
291
+ fetch("/setup/ai-credits").then((r) => r.json()).then((c) => aiCredits(c)).catch(() => {});
292
+ async function buyCredits(amountUsd) {
293
+ status("Starting checkout…");
294
+ try {
295
+ const r = await (await fetch("/setup/ai-credits/checkout", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ amountUsd }) })).json();
296
+ if (r.ok && r.url) window.open(r.url, "_blank");
297
+ else status("Checkout failed: " + (r.error || "?"));
298
+ } catch {
299
+ status("Checkout request failed.");
300
+ }
301
+ }
287
302
  const items = signal({ pages: [], posts: [] });
288
303
  const editing = signal(null); // { type, slug, body, isNew } — set only on open/save/close, so typing doesn't re-render
289
304
  const loadItems = async () => items(await (await fetch("/setup/content")).json());
@@ -339,6 +354,7 @@ const manageView = () =>
339
354
 
340
355
  const configView = () =>
341
356
  html`${() => (upgrade()?.available ? html`<div class="card-x p-3 mb-3 d-flex justify-content-between align-items-center"><span class="small">⬆ <strong>create-volt ${upgrade().latest}</strong> is available — you have ${upgrade().current}.</span><button class="btn btn-sm btn-primary" onclick=${doUpgrade}>Upgrade</button></div>` : "")}
357
+ ${() => (aiCredits()?.ok ? html`<div class="card-x p-3 mb-3"><div class="d-flex justify-content-between align-items-center mb-2"><strong>AI credits</strong><span class="small text-muted">${aiCredits().tier}${typeof aiCredits().creditBalanceUsd === "number" ? ` · $${aiCredits().creditBalanceUsd.toFixed(2)} left` : ""}</span></div>${aiCredits().payments ? html`<div class="d-flex gap-2 align-items-center"><span class="small text-muted me-1">Top up:</span>${[10, 25, 50].map((a) => html`<button class="btn btn-sm btn-outline-primary" onclick=${() => buyCredits(a)}>$${a}</button>`)}</div>` : html`<div class="small text-muted">Pay-as-you-go isn't enabled on the gateway yet — using the free tier.</div>`}</div>` : "")}
342
358
  ${available.length ? html`<div class="card-x p-4 mb-3"><h2 class="h6 mb-3">Features</h2>${available.map(addonRow)}<p class="small text-muted mb-0">Enabling a feature wires its backend automatically. Frontend UI (login form, chat) is yours to build — or start from <code>--template guestbook</code>.</p></div>` : ""}
343
359
  <div class="card-x p-4 mb-3">
344
360
  <h2 class="h6 mb-3">Settings</h2>
@@ -11,7 +11,8 @@
11
11
  "pm2": "npx --yes pm2 start ecosystem.config.cjs",
12
12
  "pm2:restart": "npx --yes pm2 restart ecosystem.config.cjs",
13
13
  "pm2:stop": "npx --yes pm2 delete ecosystem.config.cjs",
14
- "pm2:logs": "npx --yes pm2 logs"
14
+ "pm2:logs": "npx --yes pm2 logs",
15
+ "logs": "node server.js --logs"
15
16
  },
16
17
  "dependencies": {
17
18
  "express": "^4.22.2",
@@ -13,6 +13,7 @@ import fs from "node:fs";
13
13
  import path from "node:path";
14
14
  import { spawn, spawnSync } from "node:child_process";
15
15
  import { fileURLToPath, pathToFileURL } from "node:url";
16
+ import os from "node:os";
16
17
  import express from "express";
17
18
  import { Server as SocketServer } from "socket.io";
18
19
 
@@ -340,6 +341,42 @@ function startSetup() {
340
341
  }
341
342
  return;
342
343
  }
344
+ // --- AI credits: in-config purchase flow. Proxies the hosted gateway with
345
+ // the app's VOLT_AI_TOKEN — the buy flow lives here in the (shell-gated)
346
+ // config only, never in the running app. ---
347
+ if (req.method === "GET" && p === "/setup/ai-credits") {
348
+ const env = readEnvFile();
349
+ res.setHeader("Content-Type", "application/json");
350
+ if (!env.VOLT_AI_TOKEN) return res.end(JSON.stringify({ ok: false, error: "no VOLT_AI_TOKEN (BYO or unset)" }));
351
+ const base = (env.VOLT_AI_GATEWAY || "https://voltjs.com/api/ai").replace(/\/api\/ai\/?$/, "");
352
+ fetch(base + "/api/credits", { headers: { authorization: "Bearer " + env.VOLT_AI_TOKEN } })
353
+ .then((r) => r.json())
354
+ .then((j) => res.end(JSON.stringify(j)))
355
+ .catch(() => res.end(JSON.stringify({ ok: false, error: "gateway unreachable" })));
356
+ return;
357
+ }
358
+ if (req.method === "POST" && p === "/setup/ai-credits/checkout") {
359
+ let cbody = "";
360
+ req.on("data", (c) => (cbody += c));
361
+ req.on("end", () => {
362
+ const env = readEnvFile();
363
+ res.setHeader("Content-Type", "application/json");
364
+ if (!env.VOLT_AI_TOKEN) return res.end(JSON.stringify({ ok: false, error: "no VOLT_AI_TOKEN" }));
365
+ const base = (env.VOLT_AI_GATEWAY || "https://voltjs.com/api/ai").replace(/\/api\/ai\/?$/, "");
366
+ let amountUsd = 0;
367
+ try {
368
+ amountUsd = Number(JSON.parse(cbody || "{}").amountUsd) || 0;
369
+ } catch {
370
+ /* bad json */
371
+ }
372
+ const baseUrl = env.SITE_URL || `http://localhost:${configPort()}`;
373
+ fetch(base + "/api/credits/checkout", { method: "POST", headers: { "content-type": "application/json", authorization: "Bearer " + env.VOLT_AI_TOKEN }, body: JSON.stringify({ amountUsd, baseUrl }) })
374
+ .then((r) => r.json())
375
+ .then((j) => res.end(JSON.stringify(j)))
376
+ .catch(() => res.end(JSON.stringify({ ok: false, error: "gateway unreachable" })));
377
+ });
378
+ return;
379
+ }
343
380
  // --- content manager: list / read / write / delete pages + posts ---
344
381
  if (req.method === "GET" && p === "/setup/content") {
345
382
  const list = (type) => {
@@ -614,7 +651,118 @@ async function startStudio() {
614
651
  });
615
652
  }
616
653
 
617
- // --- gate: studio / setup (first run, --edit) / the app ---
654
+ // --- `--logs`: a disposable, localhost-only log viewer (its own port, like
655
+ // --studio). Tails pm2 stdout/stderr; with mir-sentinel installed, an Analytics
656
+ // tab parses Apache/nginx access logs (ACCESS_LOG). Shell access is the auth;
657
+ // for a remote box, SSH-tunnel the port. ---
658
+ async function startLogs() {
659
+ loadEnv();
660
+ const PORT = configPort();
661
+ const name = (() => {
662
+ try {
663
+ return JSON.parse(fs.readFileSync(PKG_PATH, "utf8")).name || "app";
664
+ } catch {
665
+ return "app";
666
+ }
667
+ })();
668
+ const logsDir = path.join(os.homedir(), ".pm2", "logs");
669
+ // Sources: pm2 stdout/stderr work out of the box; ACCESS_LOG adds an Apache/nginx
670
+ // log; users add more (other apps/servers/mounted or tunneled paths) via
671
+ // .volt/logs.json, editable here in the viewer. Re-read per request so additions
672
+ // show live. For a remote box, ship its log here or SSH-tunnel + run --logs there.
673
+ const LOGS_JSON = path.join(__dirname, ".volt", "logs.json");
674
+ const readExtra = () => {
675
+ try {
676
+ const a = JSON.parse(fs.readFileSync(LOGS_JSON, "utf8"));
677
+ return Array.isArray(a) ? a.filter((x) => x && x.label && x.file) : [];
678
+ } catch {
679
+ return [];
680
+ }
681
+ };
682
+ const sources = () => {
683
+ const s = { app: path.join(logsDir, `${name}-out.log`), error: path.join(logsDir, `${name}-error.log`) };
684
+ if (process.env.ACCESS_LOG) s.access = process.env.ACCESS_LOG;
685
+ for (const x of readExtra()) s[x.label] = x.file;
686
+ return s;
687
+ };
688
+ const tail = (f, n) => (f && fs.existsSync(f) ? fs.readFileSync(f, "utf8").split(/\r?\n/).filter(Boolean).slice(-n) : []);
689
+ let parseLine = null;
690
+ try {
691
+ parseLine = (await import("mir-sentinel")).parseLine;
692
+ } catch {
693
+ /* analytics optional */
694
+ }
695
+ const top = (arr, key) => {
696
+ const m = {};
697
+ for (const x of arr) if (x && x[key]) m[x[key]] = (m[x[key]] || 0) + 1;
698
+ return Object.entries(m).sort((a, b) => b[1] - a[1]).slice(0, 10);
699
+ };
700
+ const assets = {
701
+ "/": ["text/html; charset=utf-8", fs.readFileSync(path.join(__dirname, "setup", "logs.html"))],
702
+ "/logs.js": ["text/javascript; charset=utf-8", fs.readFileSync(path.join(__dirname, "setup", "logs.js"))],
703
+ };
704
+ const json = (res, o) => {
705
+ res.setHeader("Content-Type", "application/json");
706
+ res.end(JSON.stringify(o));
707
+ };
708
+ const server = http.createServer((req, res) => {
709
+ const u = new URL(req.url, "http://localhost");
710
+ const p = u.pathname;
711
+ if (assets[p]) {
712
+ res.setHeader("Content-Type", assets[p][0]);
713
+ return res.end(assets[p][1]);
714
+ }
715
+ if (p === "/api/sources") return json(res, { sources: Object.keys(sources()), extra: readExtra(), analytics: !!parseLine });
716
+ if (p === "/api/tail") {
717
+ const f = sources()[u.searchParams.get("source")];
718
+ return f ? json(res, { ok: true, lines: tail(f, Math.min(2000, Number(u.searchParams.get("lines")) || 300)) }) : json(res, { ok: false });
719
+ }
720
+ if (p === "/api/analytics") {
721
+ if (!parseLine) return json(res, { ok: false, error: "npm i mir-sentinel for analytics" });
722
+ const f = sources()[u.searchParams.get("source")];
723
+ if (!f) return json(res, { ok: false });
724
+ const parsed = tail(f, 5000).map((l) => parseLine(l));
725
+ return json(res, { ok: true, total: parsed.length, paths: top(parsed, "path"), statuses: top(parsed, "status"), ips: top(parsed, "ip"), bots: parsed.filter((x) => x && x.bot).length, attacks: parsed.filter((x) => x && x.attack).length });
726
+ }
727
+ // add/remove a source ("add servers") — written to .volt/logs.json
728
+ if (req.method === "POST" && (p === "/api/source" || p === "/api/source/remove")) {
729
+ let body = "";
730
+ req.on("data", (c) => (body += c));
731
+ req.on("end", () => {
732
+ try {
733
+ const { label, file } = JSON.parse(body || "{}");
734
+ if (!/^[a-z0-9][a-z0-9 _-]*$/i.test(label || "")) throw new Error("label: letters, numbers, spaces, - _");
735
+ let list = readExtra().filter((x) => x.label !== label);
736
+ if (p === "/api/source") list.push({ label, file: String(file || "") });
737
+ fs.mkdirSync(path.dirname(LOGS_JSON), { recursive: true });
738
+ fs.writeFileSync(LOGS_JSON, JSON.stringify(list, null, 2));
739
+ json(res, { ok: true });
740
+ } catch (e) {
741
+ res.statusCode = 400;
742
+ json(res, { ok: false, error: e.message });
743
+ }
744
+ });
745
+ return;
746
+ }
747
+ res.statusCode = 404;
748
+ res.end("not found");
749
+ });
750
+ server.on("error", (e) => {
751
+ if (e.code === "EADDRINUSE") {
752
+ console.error(`\n[volt] Logs port ${PORT} is in use — set CONFIG_PORT in .env or pass --port <n>.`);
753
+ process.exit(1);
754
+ }
755
+ throw e;
756
+ });
757
+ server.listen(PORT, "127.0.0.1", () => {
758
+ const url = `http://localhost:${PORT}`;
759
+ console.log(`\nVolt logs at ${url} (${parseLine ? "analytics on" : "raw tail — npm i mir-sentinel for analytics"})`);
760
+ console.log(" localhost only; for a remote box: ssh -L " + PORT + ":localhost:" + PORT + " you@server");
761
+ openBrowser(url);
762
+ });
763
+ }
764
+
765
+ // --- gate: studio / logs / setup (first run, --edit) / the app ---
618
766
  const editMode = process.argv.includes("--edit") || process.argv.includes("-e");
619
767
  // In production / on a PaaS there's no interactive wizard: config comes from the
620
768
  // platform's env vars (a Dockerfile sets NODE_ENV=production). Only fall back to
@@ -622,6 +770,8 @@ const editMode = process.argv.includes("--edit") || process.argv.includes("-e");
622
770
  const configured = fs.existsSync(ENV_PATH) || process.env.VOLT_ADDONS != null || process.env.NODE_ENV === "production";
623
771
  if (process.argv.includes("--studio")) {
624
772
  startStudio();
773
+ } else if (process.argv.includes("--logs")) {
774
+ startLogs();
625
775
  } else if (editMode || !configured) {
626
776
  startSetup();
627
777
  } else {
@@ -0,0 +1,29 @@
1
+ <!doctype html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="utf-8" /><meta name="viewport" content="width=device-width, initial-scale=1" />
5
+ <title>Logs — Volt</title><meta name="robots" content="noindex" /><link rel="icon" href="/favicon.webp" />
6
+ <link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.3/dist/css/bootstrap.min.css" rel="stylesheet" />
7
+ <style>body{background:#0f1115;color:#e7e9ee}pre{background:#0b0d11;color:#cfe3ff;border:1px solid #232a36;border-radius:8px;padding:12px;max-height:72vh;overflow:auto;font-size:12.5px;white-space:pre-wrap}</style>
8
+ </head>
9
+ <body>
10
+ <div class="container-fluid py-3">
11
+ <div class="d-flex gap-2 align-items-center mb-2 flex-wrap">
12
+ <strong class="me-2">Volt logs</strong>
13
+ <select id="src" class="form-select form-select-sm" style="max-width:200px"></select>
14
+ <select id="view" class="form-select form-select-sm" style="max-width:160px"><option value="tail">Raw tail</option><option value="analytics">Analytics</option></select>
15
+ <button id="refresh" class="btn btn-sm btn-outline-secondary">Refresh</button>
16
+ <button id="toggleadd" class="btn btn-sm btn-outline-secondary" title="Add another log source (server, app, mounted/tunneled path)">+ source</button>
17
+ <label class="form-check-label small ms-1"><input id="follow" type="checkbox" class="form-check-input" /> follow</label>
18
+ <input id="filter" class="form-control form-control-sm ms-auto" style="max-width:240px" placeholder="filter…" />
19
+ </div>
20
+ <div id="addbar" class="d-flex gap-2 mb-2" style="display:none">
21
+ <input id="newlabel" class="form-control form-control-sm" style="max-width:160px" placeholder="label (e.g. prod)" />
22
+ <input id="newfile" class="form-control form-control-sm" style="max-width:420px" placeholder="/var/log/nginx/access.log (local, mounted, or shipped path)" />
23
+ <button id="addsrc" class="btn btn-sm btn-primary">Add</button>
24
+ </div>
25
+ <div id="out"></div>
26
+ </div>
27
+ <script type="module" src="/logs.js"></script>
28
+ </body>
29
+ </html>
@@ -0,0 +1,58 @@
1
+ // logs.js — the --logs viewer client (raw tail + mir-sentinel analytics).
2
+ const $ = (s) => document.querySelector(s);
3
+ const esc = (s) => String(s).replace(/[&<>]/g, (c) => ({ "&": "&amp;", "<": "&lt;", ">": "&gt;" })[c]);
4
+ let timer = null;
5
+
6
+ async function loadSources() {
7
+ const { sources = [] } = await (await fetch("/api/sources")).json();
8
+ $("#src").innerHTML = sources.map((s) => `<option value="${esc(s)}">${esc(s)}</option>`).join("") || `<option>(no logs found)</option>`;
9
+ }
10
+
11
+ async function render() {
12
+ const src = $("#src").value;
13
+ if (!src) return;
14
+ if ($("#view").value === "analytics") {
15
+ const a = await (await fetch(`/api/analytics?source=${encodeURIComponent(src)}`)).json();
16
+ if (!a.ok) {
17
+ $("#out").innerHTML = `<div class="text-muted small">${esc(a.error || "no analytics")}</div>`;
18
+ return;
19
+ }
20
+ const tbl = (title, rows) => `<h6 class="mt-3">${title}</h6><table class="table table-dark table-sm mb-0"><tbody>${(rows || []).map(([k, v]) => `<tr><td>${esc(String(k))}</td><td class="text-end">${v}</td></tr>`).join("")}</tbody></table>`;
21
+ $("#out").innerHTML = `<div class="small text-muted mb-2">${a.total} lines · ${a.bots} bot · ${a.attacks} attack</div>` + tbl("Top paths", a.paths) + tbl("Status codes", a.statuses) + tbl("Top IPs", a.ips);
22
+ } else {
23
+ const { lines = [] } = await (await fetch(`/api/tail?source=${encodeURIComponent(src)}&lines=400`)).json();
24
+ const filter = $("#filter").value.toLowerCase();
25
+ const shown = filter ? lines.filter((l) => l.toLowerCase().includes(filter)) : lines;
26
+ const pre = document.createElement("pre");
27
+ pre.textContent = shown.join("\n") || "(empty)";
28
+ $("#out").innerHTML = "";
29
+ $("#out").appendChild(pre);
30
+ pre.scrollTop = pre.scrollHeight;
31
+ }
32
+ }
33
+
34
+ $("#toggleadd").onclick = () => {
35
+ const b = $("#addbar");
36
+ b.style.display = b.style.display === "none" ? "flex" : "none";
37
+ };
38
+ $("#addsrc").onclick = async () => {
39
+ const label = $("#newlabel").value.trim();
40
+ const file = $("#newfile").value.trim();
41
+ if (!label || !file) return;
42
+ const r = await (await fetch("/api/source", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ label, file }) })).json();
43
+ if (!r.ok) return alert(r.error || "could not add source");
44
+ $("#newlabel").value = $("#newfile").value = "";
45
+ $("#addbar").style.display = "none";
46
+ await loadSources();
47
+ $("#src").value = label;
48
+ render();
49
+ };
50
+ $("#refresh").onclick = render;
51
+ $("#src").onchange = render;
52
+ $("#view").onchange = render;
53
+ $("#filter").oninput = render;
54
+ $("#follow").onchange = (e) => {
55
+ clearInterval(timer);
56
+ if (e.target.checked) timer = setInterval(render, 3000);
57
+ };
58
+ loadSources().then(render);