Use page origin for share links except when browsing at localhost
The LAN-IP rewrite in _getShareBaseUrl() exists to fix unusable 127.0.0.1 links; applying it to every origin meant links copied behind a reverse proxy pointed at http://<LAN-IP>:5100, bypassing TLS. HTTPS and non-localhost origins are now used as-is. Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
This commit is contained in:
parent
c79e7097ea
commit
679f91da2c
@ -9,6 +9,10 @@ Version numbers follow [Semantic Versioning](https://semver.org/spec/v2.0.0.html
|
|||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- **Share links now respect a reverse proxy** — `_getShareBaseUrl()` rewrote every copied share link to `http://<LAN-IP>:5100` (via `/api/local_ip`), which would bypass TLS when the scanner sits behind a reverse proxy (Zoraxy, Caddy, nginx, …): a DPO opening the link would silently fall back to plain HTTP. The LAN-IP rewrite now only applies in the case it was built for — browsing the app at `localhost` over HTTP, where `window.location.origin` would produce links unusable from other machines. Any HTTPS or non-localhost origin is used as-is.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## [1.7.2] — 2026-06-10
|
## [1.7.2] — 2026-06-10
|
||||||
|
|||||||
@ -65,7 +65,7 @@ Never revert to `!!window._googleConnected` / `_fileSources.length > 0` — thos
|
|||||||
- **`window.VIEWER_MODE`** — injected by Jinja2. `auth.js` adds `viewer-mode` class to `<body>`; all hide rules are CSS (`body.viewer-mode …`) except `delBtn` which is also guarded in JS.
|
- **`window.VIEWER_MODE`** — injected by Jinja2. `auth.js` adds `viewer-mode` class to `<body>`; all hide rules are CSS (`body.viewer-mode …`) except `delBtn` which is also guarded in JS.
|
||||||
- **`window.VIEWER_SCOPE`** — injected alongside `VIEWER_MODE`. If `VIEWER_SCOPE.role` is set, `auth.js` pre-sets `#filterRole` and hides the dropdown.
|
- **`window.VIEWER_SCOPE`** — injected alongside `VIEWER_MODE`. If `VIEWER_SCOPE.role` is set, `auth.js` pre-sets `#filterRole` and hides the dropdown.
|
||||||
- **Token onclick attributes** — Copy/Revoke buttons pass the token as a single-quoted JS string literal, never via `JSON.stringify` (which produces double-quoted strings that break `onclick="…"` attributes).
|
- **Token onclick attributes** — Copy/Revoke buttons pass the token as a single-quoted JS string literal, never via `JSON.stringify` (which produces double-quoted strings that break `onclick="…"` attributes).
|
||||||
- **Share link base URL** — `_getShareBaseUrl()` fetches `/api/local_ip` (LAN IP via UDP probe to `8.8.8.8`) so copied links are routable from other machines. Both `createShareLink` and `copyTokenLink` are `async`. Do not revert to `window.location.origin` — that produces `127.0.0.1` links.
|
- **Share link base URL** — `_getShareBaseUrl()` uses `window.location.origin` whenever the page is served over HTTPS or from a non-localhost host (a reverse-proxied hostname or LAN IP is already routable, and rewriting it to `http://<LAN-IP>` would bypass the proxy's TLS). Only when browsing at `localhost`/`127.0.0.1` over HTTP does it fetch `/api/local_ip` (LAN IP via UDP probe to `8.8.8.8`) so copied links work from other machines. The result is cached in `_shareBaseUrl` so Copy buttons stay within the click gesture. Both `createShareLink` and `copyTokenLink` are `async`. Do not make it return bare `window.location.origin` unconditionally — that reintroduces unusable `127.0.0.1` links.
|
||||||
- **Settings Security pane** — Admin PIN and Viewer PIN groups live in `stPaneSecurity`. `switchSettingsTab('security')` triggers both `stLoadPinStatus()` and `stLoadViewerPinStatus()`.
|
- **Settings Security pane** — Admin PIN and Viewer PIN groups live in `stPaneSecurity`. `switchSettingsTab('security')` triggers both `stLoadPinStatus()` and `stLoadViewerPinStatus()`.
|
||||||
|
|
||||||
## Gotchas
|
## Gotchas
|
||||||
|
|||||||
@ -5,8 +5,17 @@ import { S } from './state.js';
|
|||||||
let _shareBaseUrl = null; // cached so Copy buttons can build the URL synchronously
|
let _shareBaseUrl = null; // cached so Copy buttons can build the URL synchronously
|
||||||
|
|
||||||
async function _getShareBaseUrl() {
|
async function _getShareBaseUrl() {
|
||||||
// Use the machine's LAN IP so links work for remote users, not just localhost.
|
|
||||||
if (_shareBaseUrl) return _shareBaseUrl;
|
if (_shareBaseUrl) return _shareBaseUrl;
|
||||||
|
// The LAN-IP probe exists only to fix links when the operator browses the
|
||||||
|
// app at localhost — those would be unusable for remote users. Any other
|
||||||
|
// origin (LAN IP, or a reverse-proxied HTTPS hostname) is already routable,
|
||||||
|
// and rewriting it to http://<LAN-IP> would bypass the proxy's TLS.
|
||||||
|
const host = window.location.hostname;
|
||||||
|
if (window.location.protocol === 'https:' ||
|
||||||
|
(host !== 'localhost' && host !== '127.0.0.1' && host !== '[::1]')) {
|
||||||
|
_shareBaseUrl = window.location.origin;
|
||||||
|
return _shareBaseUrl;
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
const r = await fetch('/api/local_ip');
|
const r = await fetch('/api/local_ip');
|
||||||
if (r.ok) {
|
if (r.ok) {
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user