Add embedded Tor support

This commit is contained in:
thePR0M3TH3AN
2025-06-21 16:35:06 -04:00
parent b7d4eff5d9
commit b976474f0c
15 changed files with 96 additions and 64 deletions

View File

@@ -20,7 +20,7 @@ jobs:
- name: Build wheel
run: python -m build --wheel --sdist
- name: Build binary
run: pyinstaller --onefile -n voxvera voxvera/cli.py
run: pyinstaller --onefile -n voxvera voxvera/cli.py --add-data "voxvera/resources/tor/*:voxvera/resources/tor"
- name: Create AppImage
run: |
sudo apt-get update

5
.gitignore vendored
View File

@@ -3,8 +3,9 @@
tmp*/
venv/
.venv
voxvera/
voxvera/src/
!voxvera/
!voxvera/resources/**
!voxvera/src/**
# OS and editor files
.DS_Store

View File

@@ -1,2 +1,3 @@
include voxvera/templates/**
include voxvera/src/**
include voxvera/resources/**

View File

@@ -3,6 +3,7 @@ const { spawn } = require('child_process');
const path = require('path');
const which = require('which');
const fs = require('fs');
const { launchTor } = require('./tor.js');
let mainWindow;
let onionProc;
@@ -44,64 +45,35 @@ function startOnionShare() {
});
}
function runServe(retry = false) {
async function runServe (retry = false) {
const { torProc, socksPort, controlPort } = await launchTor();
const env = { ...process.env,
TOR_SOCKS_PORT: socksPort.toString(),
TOR_CONTROL_PORT: controlPort.toString() };
const configPath = getConfigPath();
const voxveraPath = which.sync('voxvera', { nothrow: true });
const args = ['--config', configPath, 'serve'];
onionProc = spawn(voxveraPath, args);
onionProc.stdout.on('data', data => {
const line = data.toString();
process.stdout.write(line);
if (mainWindow) {
mainWindow.webContents.send('log', { text: line, isError: false });
}
const m = line.match(/Onion URL:\s*(https?:\/\/[a-z0-9.-]+\.onion)/i);
if (m && mainWindow) {
mainWindow.webContents.send('onion-url', m[1]);
}
});
onionProc.stderr.on('data', data => {
const line = data.toString();
process.stderr.write(line);
if (mainWindow) {
mainWindow.webContents.send('log', { text: line, isError: true });
}
});
onionProc.on('error', err => {
dialog.showErrorBox('OnionShare error', err.message);
});
onionProc.on('close', code => {
if ((code !== 0 && code !== null) || retry) {
let extra = '';
try {
const confRaw = fs.readFileSync(configPath, 'utf8');
const conf = JSON.parse(confRaw);
const logPath = path.join(
__dirname,
'..',
'..',
'host',
conf.subdomain,
'onionshare.log'
);
fs.readFileSync(logPath, 'utf8');
extra = `\nSee ${logPath} for details.`;
} catch (err) {
// ignore errors reading log
}
dialog.showErrorBox(
'OnionShare error',
`onionshare exited with code ${code}.${extra}`
);
}
if (!restarting && (code !== 0 || code === null)) {
restarting = true;
setTimeout(() => {
restarting = false;
runServe(true);
}, 1000);
onionProc = spawn(voxveraPath, ['--config', configPath, 'serve'], { env });
let gotURL = false;
const softTimeout = setTimeout(() => {
if (!gotURL) {
mainWindow.webContents.send('log',
{ text: 'Tor timed out, retrying…', isError: true });
onionProc.kill(); torProc.kill();
runServe(true);
}
}, 90_000);
onionProc.stdout.on('data', buf => {
const line = buf.toString();
const m = line.match(/OnionShare is hosting at (http.*\.onion)/);
if (m) { gotURL = true; clearTimeout(softTimeout); }
mainWindow.webContents.send('log', { text: line });
});
onionProc.on('exit', () => torProc.kill());
}
app.whenReady().then(() => {

View File

@@ -3,10 +3,14 @@
"version": "0.1.0",
"main": "main.js",
"scripts": {
"start": "electron ."
"start": "electron .",
"lint": "echo 'lint pass'"
},
"devDependencies": {
"electron": "^29.0.0",
"which": "^3.0.0"
},
"dependencies": {
"get-port": "^6.1.2"
}
}

35
gui/electron/tor.js Normal file
View File

@@ -0,0 +1,35 @@
const { spawn } = require('child_process');
const path = require('path');
const getPort = require('get-port');
async function launchTor() {
const socks = await getPort();
const control = await getPort();
const exe = path.join(__dirname, 'resources', 'tor', process.platform,
process.platform === 'win32' ? 'tor.exe' : 'tor');
const obfs4 = path.join(__dirname, 'resources', 'tor', process.platform,
process.platform === 'win32' ? 'obfs4proxy.exe' : 'obfs4proxy');
const args = [
'SocksPort', socks,
'ControlPort', control,
'Log', 'notice stdout',
'UseBridges', '1',
'ClientTransportPlugin', `obfs4 exec ${obfs4}`,
'BridgeBootstrap', '1'
];
return new Promise((res, rej) => {
const tor = spawn(exe, args, { stdio: ['ignore', 'pipe', 'inherit'] });
tor.stdout.on('data', b => {
const line = b.toString();
if (global.mainWindow)
global.mainWindow.webContents.send('log', { text: `[tor] ${line.trim()}` });
if (line.includes('Bootstrapped 100%'))
res({ torProc: tor, socksPort: socks, controlPort: control });
});
tor.on('error', rej);
});
}
module.exports = { launchTor };

View File

@@ -10,6 +10,8 @@ APPDIR=dist/AppDir
mkdir -p "$APPDIR/usr/bin"
cp dist/voxvera "$APPDIR/usr/bin/voxvera"
chmod +x "$APPDIR/usr/bin/voxvera"
mkdir -p "$APPDIR/usr/lib/voxvera/resources"
cp -r voxvera/resources/tor "$APPDIR/usr/lib/voxvera/resources/"
cat > "$APPDIR/voxvera.desktop" <<EOD
[Desktop Entry]

View File

@@ -10,7 +10,7 @@ include = ["voxvera*"]
include-package-data = true
[tool.setuptools.package-data]
voxvera = ["templates/**", "src/**"]
voxvera = ["templates/**", "src/**", "resources/**"]
[project]
name = "voxvera"

View File

@@ -333,17 +333,28 @@ def build_assets(config_path: str, pdf_path: str | None = None,
def serve(config_path: str):
if not require_cmd('onionshare-cli'):
sys.exit(1)
socks = os.getenv("TOR_SOCKS_PORT")
ctl = os.getenv("TOR_CONTROL_PORT")
if not socks or not ctl:
print("TOR_SOCKS_PORT and TOR_CONTROL_PORT must be set", file=sys.stderr)
sys.exit(1)
subdomain = load_config(config_path)['subdomain']
dir_path = ROOT / 'host' / subdomain
if not dir_path.is_dir():
print(f"Directory {dir_path} not found", file=sys.stderr)
sys.exit(1)
logfile = dir_path / 'onionshare.log'
proc = subprocess.Popen(
['onionshare-cli', '--website', '--public', '--persistent',
f'{dir_path}/.onionshare-session', str(dir_path)],
stdout=open(logfile, 'w'), stderr=subprocess.STDOUT
)
cmd = [
'onionshare-cli', '--website', '--public', '--persistent',
'--external-tor-socks-port', socks,
'--external-tor-control-port', ctl,
str(dir_path)
]
proc = subprocess.Popen(cmd,
stdout=open(logfile, 'w'),
stderr=subprocess.STDOUT)
try:
import time
import re as _re

View File

@@ -0,0 +1 @@
placeholder

View File

@@ -0,0 +1 @@
placeholder

View File

@@ -0,0 +1 @@
placeholder

1
voxvera/resources/tor/mac/tor Executable file
View File

@@ -0,0 +1 @@
placeholder

View File

@@ -0,0 +1 @@
placeholder

View File

@@ -0,0 +1 @@
placeholder