monerod-gui/app/main.ts

1065 lines
30 KiB
TypeScript
Raw Normal View History

2024-10-23 21:13:31 +00:00
import { app, BrowserWindow, ipcMain, screen, dialog, Tray, Menu, MenuItemConstructorOptions,
IpcMainInvokeEvent, Notification, NotificationConstructorOptions, clipboard
2024-10-23 21:13:31 +00:00
} from 'electron';
import { ChildProcessWithoutNullStreams, exec, ExecException, spawn } from 'child_process';
2024-09-17 18:33:24 +00:00
import * as path from 'path';
import * as fs from 'fs';
2024-09-28 14:31:51 +00:00
import * as https from 'https';
import { createHash } from 'crypto';
import * as tar from 'tar';
2024-10-09 21:40:32 +00:00
import * as os from 'os';
2024-10-23 21:13:31 +00:00
import AutoLaunch from './auto-launch';
2024-11-10 15:57:02 +00:00
import AdmZip from 'adm-zip';
2024-11-08 17:36:05 +00:00
2024-11-10 15:57:02 +00:00
const pidusage = require('pidusage');
2024-11-08 17:36:05 +00:00
const batteryLevel = require('battery-level');
2024-11-01 20:45:25 +00:00
const network = require('network');
2024-10-21 20:36:05 +00:00
2024-10-16 16:18:43 +00:00
interface Stats {
/**
* percentage (from 0 to 100*vcore)
*/
cpu: number;
/**
* bytes
*/
memory: number;
/**
* PPID
*/
ppid: number;
/**
* PID
*/
pid: number;
/**
* ms user + system time
*/
ctime: number;
/**
* ms since the start of the process
*/
elapsed: number;
/**
* ms since epoch
*/
timestamp: number;
}
2024-10-09 21:40:32 +00:00
2024-09-29 22:29:16 +00:00
//import bz2 from 'unbzip2-stream';
//import * as bz2 from 'unbzip2-stream';
const bz2 = require('unbzip2-stream');
2024-09-22 13:55:03 +00:00
app.setName('Monero Daemon');
2024-10-17 16:52:17 +00:00
let autoLauncher = new AutoLaunch({
2024-10-23 21:13:31 +00:00
name: 'monerod-gui',
path: process.execPath,
options: {
extraArguments: [
'--auto-launch'
2024-10-29 14:42:58 +00:00
],
linux: {
comment: 'Monerod GUI startup script',
2024-11-02 13:14:54 +00:00
version: '0.1.2'
2024-10-29 14:42:58 +00:00
}
2024-10-23 21:13:31 +00:00
}
2024-10-21 20:36:05 +00:00
});
2024-10-23 17:01:24 +00:00
const isAutoLaunched: boolean = process.argv.includes('--auto-launch');
const minimized: boolean = process.argv.includes('--hidden');
2024-10-23 17:01:24 +00:00
2024-09-17 18:33:24 +00:00
let win: BrowserWindow | null = null;
2024-10-12 14:36:15 +00:00
let isHidden: boolean = false;
2024-10-12 12:28:16 +00:00
let isQuitting: boolean = false;
2024-11-01 20:45:25 +00:00
const separator: string = os.platform() == 'win32' ? '\\' : '/';
const appApp = `${separator}app${separator}app`;
const appSrc = `${separator}app${separator}src`;
const _app = `${separator}app`;
const _src = `${separator}src`;
const dirname = (__dirname.endsWith(appApp) ? __dirname.replace(appApp, appSrc) : __dirname.endsWith(_app) ? __dirname.replace(_app, _src) : __dirname);
console.log('dirname: ' + dirname);
2024-10-16 16:18:43 +00:00
let monerodProcess: ChildProcessWithoutNullStreams | null = null;
2024-10-23 15:17:33 +00:00
const iconRelPath: string = 'assets/icons/monero-symbol-on-white-480.png';
2024-11-01 20:45:25 +00:00
//const wdwIcon = `${dirname}/${iconRelPath}`;
const wdwIcon = path.join(dirname, iconRelPath);
2024-10-12 12:28:16 +00:00
2024-10-31 19:35:47 +00:00
let tray: Tray;
let trayMenu: Menu;
2024-09-17 18:33:24 +00:00
const args = process.argv.slice(1),
serve = args.some(val => val === '--serve');
const isAppImage: () => boolean = () => {
return (!!process.env.APPIMAGE) || (!!process.env.PORTABLE_EXECUTABLE_DIR);
}
2024-10-31 19:35:47 +00:00
2024-10-31 11:37:41 +00:00
// #region Window
2024-11-01 13:39:47 +00:00
function updateTrayMenu(): void {
tray.setContextMenu(trayMenu);
}
2024-09-17 18:33:24 +00:00
2024-11-01 13:39:47 +00:00
function setTrayItemEnabled(id: string, enabled: boolean): void {
const item = trayMenu.getMenuItemById(id);
if (!item) {
throw new Error(`Item not found: ${id}`);
}
item.enabled = enabled;
updateTrayMenu();
}
2024-10-31 19:35:47 +00:00
function createTrayMenuTemplate(): MenuItemConstructorOptions[] {
return [
2024-10-12 12:28:16 +00:00
{
2024-11-01 13:39:47 +00:00
id: "startDaemon",
2024-10-31 19:35:47 +00:00
label: "Start",
2024-11-01 13:39:47 +00:00
toolTip: "Start Daemon",
click: () => {
win?.webContents.send('on-tray-start-daemon');
}
},
{
id: "stopDaemon",
label: "Stop",
toolTip: "Stop Daemon",
click: () => {
win?.webContents.send('on-tray-stop-daemon');
}
},
{
id: "startSync",
label: "Start Sync",
toolTip: "Start Daemon Sync",
click: () => {
win?.webContents.send('on-tray-start-sync');
}
},
{
id: "stopSync",
label: "Stop Sync",
toolTip: "Stop Daemon Sync",
click: () => {
win?.webContents.send('on-tray-stop-sync');
}
2024-10-12 12:28:16 +00:00
},
{
id: "quitDaemon",
label: "Quit",
2024-11-01 13:39:47 +00:00
toolTip: "Quit Daemon",
2024-10-12 12:28:16 +00:00
click: () => {
2024-11-01 13:39:47 +00:00
win?.webContents.send('on-tray-quit-daemon');
2024-10-12 12:28:16 +00:00
}
}
2024-10-31 19:35:47 +00:00
]
}
2024-10-12 12:28:16 +00:00
2024-10-31 19:35:47 +00:00
function createTray(): Tray {
const trayMenuTemplate = createTrayMenuTemplate();
2024-10-12 12:28:16 +00:00
const tray = new Tray(wdwIcon);
2024-10-31 19:35:47 +00:00
trayMenu = Menu.buildFromTemplate(trayMenuTemplate);
2024-10-12 14:36:15 +00:00
tray.setToolTip('Monero Daemon');
2024-10-12 12:28:16 +00:00
tray.setContextMenu(trayMenu);
2024-10-31 19:35:47 +00:00
tray.on('click', (event) => {
if (isHidden) {
win?.show();
isHidden = false;
}
else
{
win?.hide();
isHidden = true;
}
});
return tray;
}
function createWindow(): BrowserWindow {
const size = screen.getPrimaryDisplay().workAreaSize;
tray = createTray();
2024-09-17 18:33:24 +00:00
// Create the browser window.
win = new BrowserWindow({
x: 0,
y: 0,
width: size.width,
height: size.height,
webPreferences: {
preload: path.join(__dirname, 'preload.js'),
nodeIntegration: false,
allowRunningInsecureContent: (serve),
contextIsolation: true,
2024-10-17 16:52:17 +00:00
devTools: !app.isPackaged,
2024-11-10 15:57:02 +00:00
sandbox: true
2024-09-17 18:33:24 +00:00
},
2024-10-29 17:34:25 +00:00
show: false,
2024-10-09 18:53:38 +00:00
autoHideMenuBar: true,
2024-10-12 12:28:16 +00:00
icon: wdwIcon
});
2024-09-17 18:33:24 +00:00
isHidden = minimized;
if (!app.isPackaged) win.webContents.openDevTools();
2024-10-07 21:47:52 +00:00
2024-09-17 18:33:24 +00:00
if (serve) {
const debug = require('electron-debug');
debug();
require('electron-reloader')(module);
win.loadURL('http://localhost:4200');
} else {
// Path when running electron executable
let pathIndex = './index.html';
2024-10-23 15:17:33 +00:00
if (fs.existsSync(path.join(dirname, '../dist/index.html'))) {
2024-09-17 18:33:24 +00:00
// Path when running electron in local folder
pathIndex = '../dist/index.html';
}
2024-10-23 15:17:33 +00:00
const url = new URL(path.join('file:', dirname, pathIndex));
console.log(`Main window url: ${url}`);
2024-09-17 18:33:24 +00:00
win.loadURL(url.href);
}
2024-10-12 12:28:16 +00:00
win.on('close', (event) => {
if (!isQuitting) {
event.preventDefault();
win?.hide();
2024-10-12 14:36:15 +00:00
isHidden = true;
2024-10-12 12:28:16 +00:00
//event.returnValue = false;
}
return false;
});
2024-09-17 18:33:24 +00:00
// Emitted when the window is closed.
win.on('closed', () => {
// Dereference the window object, usually you would store window
// in an array if your app supports multi windows, this is the time
// when you should delete the corresponding element.
win = null;
});
return win;
}
2024-11-08 15:14:54 +00:00
const createSplashWindow = async (): Promise<BrowserWindow | undefined> => {
return undefined;
if (os.platform() == 'win32' || isAppImage()) {
return undefined;
}
2024-11-01 20:45:25 +00:00
2024-10-29 17:34:25 +00:00
const window = new BrowserWindow({
width: 480,
height: 480,
transparent: true,
frame: false,
alwaysOnTop: true,
show: false,
icon: wdwIcon,
minimizable: false,
maximizable: false,
fullscreen: false
});
// Path when running electron executable
let pathIndex = './splash.html';
if (fs.existsSync(path.join(dirname, '../dist/splash.html'))) {
// Path when running electron in local folder
pathIndex = '../dist/splash.html';
}
const url = new URL(path.join('file:', dirname, pathIndex));
2024-10-29 17:34:25 +00:00
await window.loadURL(url.href);
await new Promise<void>((resolve) => {
setTimeout(() => {
window.show();
resolve();
}, 400);
2024-10-29 17:34:25 +00:00
});
return window;
}
2024-10-21 20:36:05 +00:00
// #endregion
// #region WiFi
function isConnectedToWiFi(): Promise<boolean> {
2024-11-01 20:45:25 +00:00
return new Promise<boolean>((resolve, reject) => {
network.get_active_interface((err: any | null, obj: { name: string, ip_address: string, mac_address: string, type: string, netmask: string, gateway_ip: string }) => {
if (err) {
console.error("Errore durante il controllo della connessione Wi-Fi:", err);
reject(err);
}
else {
console.log('isConnectedToWifi:');
console.log(obj);
resolve(obj.type == 'Wireless');
}
})
});
}
function isConnectedToWiFiOld(): Promise<boolean> {
return new Promise((resolve, reject) => {
const platform = os.platform(); // Use os to get the platform
let command = '';
if (platform === 'win32') {
// Windows: Use 'netsh' command to check the Wi-Fi status
command = 'netsh wlan show interfaces';
} else if (platform === 'darwin') {
// macOS: Use 'airport' command to check the Wi-Fi status
command = "/System/Library/PrivateFrameworks/Apple80211.framework/Versions/Current/Resources/airport -I | grep 'state: running'";
} else if (platform === 'linux') {
// Linux: Use 'nmcli' to check for Wi-Fi connectivity
command = 'nmcli dev status';
} else {
resolve(false); // Unsupported platform
}
// Execute the platform-specific command
if (command) {
exec(command, (error: ExecException | null, stdout: string, stderr: string) => {
if (error) {
console.error(error);
reject(stderr);
resolve(false); // In case of error, assume not connected to Wi-Fi
} else {
// Check if the output indicates a connected status
if (stdout) {
const components: string[] = stdout.split("\n");
components.forEach((component: string) => {
if (component.includes('wifi') && !component.includes('--')) {
resolve(true);
}
});
resolve(false);
} else {
resolve(false);
}
}
});
}
});
}
function isWifiConnected() {
isConnectedToWiFi().then((connected: boolean) => {
win?.webContents.send('is-wifi-connected-result', connected);
}).catch((error: any) => {
console.error(error);
win?.webContents.send('is-wifi-connected-result', false);
});
}
2024-10-21 20:36:05 +00:00
// #endregion
// #region monerod
2024-09-26 16:45:28 +00:00
function getMonerodVersion(monerodFilePath: string): void {
2024-10-09 19:44:37 +00:00
const monerodProcess = spawn(monerodFilePath, [ '--version' ]);
2024-09-26 16:45:28 +00:00
monerodProcess.stdout.on('data', (data) => {
2024-10-01 18:45:10 +00:00
win?.webContents.send('monero-version', `${data}`);
2024-09-26 16:45:28 +00:00
})
monerodProcess.stderr.on('data', (data) => {
2024-10-01 18:45:10 +00:00
win?.webContents.send('monero-version-error', `${data}`);
2024-09-26 16:45:28 +00:00
})
}
function checkValidMonerodPath(monerodPath: string): void {
let foundUsage: boolean = false;
const monerodProcess = spawn(monerodPath, ['--help']);
monerodProcess.stderr.on('data', (data) => {
win?.webContents.send('on-check-valid-monerod-path', false);
});
monerodProcess.stdout.on('data', (data) => {
if (`${data}`.includes('monerod [options|settings] [daemon_command...]')) {
foundUsage = true;
}
});
monerodProcess.on('close', (code: number) => {
win?.webContents.send('on-check-valid-monerod-path', foundUsage);
})
}
let moneroFirstStdout: boolean = true;
2024-09-26 16:45:28 +00:00
function startMoneroDaemon(commandOptions: string[]): ChildProcessWithoutNullStreams {
2024-10-03 20:35:30 +00:00
const monerodPath = commandOptions.shift();
if (!monerodPath) {
const error = `Invalid monerod path provided: ${monerodPath}`;
win?.webContents.send('monero-stderr', error);
2024-10-16 16:18:43 +00:00
throw new Error("Invalid monerod path provided");
}
if (monerodProcess != null) {
const error: string = 'Monero daemon already started';
win?.webContents.send('monero-stderr', error);
2024-10-16 16:18:43 +00:00
throw new Error("Monerod already started");
2024-10-03 20:35:30 +00:00
}
const message: string = "Starting monerod daemon with options: " + commandOptions.join(" ");
console.log(message);
moneroFirstStdout = true;
2024-09-22 13:55:03 +00:00
2024-11-01 20:45:25 +00:00
commandOptions.push('--non-interactive');
2024-09-22 13:55:03 +00:00
// Avvia il processo usando spawn
2024-10-16 16:18:43 +00:00
monerodProcess = spawn(monerodPath, commandOptions);
2024-09-22 13:55:03 +00:00
// Gestisci l'output di stdout in streaming
monerodProcess.stdout.on('data', (data) => {
2024-09-26 16:45:28 +00:00
//console.log(`monerod stdout: ${data}`);
const pattern = '**********************************************************************';
if (moneroFirstStdout && data.includes(pattern)) {
win?.webContents.send('monerod-started', true);
moneroFirstStdout = false;
}
2024-09-25 21:24:46 +00:00
win?.webContents.send('monero-stdout', `${data}`);
2024-09-22 13:55:03 +00:00
// Puoi anche inviare i log all'interfaccia utente tramite IPC
});
// Gestisci gli errori in stderr
monerodProcess.stderr.on('data', (data) => {
console.error(`monerod error: ${data}`);
if (moneroFirstStdout) {
win?.webContents.send('monerod-started', false);
moneroFirstStdout = false;
}
2024-09-25 21:24:46 +00:00
win?.webContents.send('monero-stderr', `${data}`);
2024-09-22 13:55:03 +00:00
});
// Gestisci la chiusura del processo
monerodProcess.on('close', (code: number) => {
console.log(`monerod exited with code: ${code}`);
2024-09-25 21:24:46 +00:00
win?.webContents.send('monero-stdout', `monerod exited with code: ${code}`);
2024-09-28 14:31:51 +00:00
win?.webContents.send('monero-close', code);
2024-10-16 16:18:43 +00:00
monerodProcess = null;
2024-09-22 13:55:03 +00:00
});
return monerodProcess;
}
2024-10-16 16:18:43 +00:00
function monitorMonerod(): void {
if (!monerodProcess) {
win?.webContents.send('on-monitor-monerod-error', 'Monerod not running');
return;
}
if (!monerodProcess.pid) {
win?.webContents.send('on-monitor-monerod-error', 'Unknown monero pid');
return;
}
pidusage(monerodProcess.pid, (error: Error | null, stats: Stats) => {
if (error) {
win?.webContents.send('on-monitor-monerod-error', `${error}`);
return;
}
win?.webContents.send('on-monitor-monerod', stats);
});
}
2024-10-21 20:36:05 +00:00
// #endregion
// #region Download Utils
2024-09-29 22:29:16 +00:00
const downloadFile = (url: string, destinationDir: string, onProgress: (progress: number) => void): Promise<string> => {
return new Promise((resolve, reject) => {
const request = (url: string) => {
https.get(url, (response) => {
if (response.statusCode === 200) {
const contentDisposition = response.headers['content-disposition'];
let finalFilename = '';
// Estrai il nome del file dall'URL o dal content-disposition
if (contentDisposition && contentDisposition.includes('filename')) {
const match = contentDisposition.match(/filename="(.+)"/);
if (match) {
finalFilename = match[1];
}
} else {
// Se non c'è content-disposition, prendiamo il nome dall'URL
finalFilename = url.split('/').pop() || 'downloaded-file';
}
const destination = `${destinationDir}/${finalFilename}`;
const file = fs.createWriteStream(destination);
const totalBytes = parseInt(response.headers['content-length'] || '0', 10);
let downloadedBytes = 0;
response.on('data', (chunk) => {
downloadedBytes += chunk.length;
const progress = (downloadedBytes / totalBytes) * 100;
onProgress(progress); // Notifica il progresso
});
response.pipe(file);
file.on('finish', () => {
file.close(() => resolve(finalFilename)); // Restituisci il nome del file finale
});
} else if (response.statusCode === 301 || response.statusCode === 302) {
// Se è un redirect, effettua una nuova richiesta verso il location header
const newUrl = response.headers.location;
if (newUrl) {
request(newUrl); // Ripeti la richiesta con il nuovo URL
} else {
reject(new Error('Redirection failed without a location header'));
}
} else {
reject(new Error(`Failed to download: ${response.statusCode}`));
}
}).on('error', (err) => {
reject(err);
});
};
request(url); // Inizia la richiesta
});
};
2024-09-28 14:31:51 +00:00
// Funzione per scaricare e verificare l'hash
const downloadAndVerifyHash = async (hashUrl: string, fileName: string, filePath: string): Promise<boolean> => {
2024-09-29 22:29:16 +00:00
//const hashFilePath = path.join(app.getPath('temp'), 'monero_hashes.txt');
2024-09-28 14:31:51 +00:00
// Scarica il file di hash
2024-09-29 22:29:16 +00:00
const hashFileName = await downloadFile(hashUrl, app.getPath('temp'), () => {});
const hashFilePath = `${app.getPath('temp')}/${hashFileName}`;
2024-09-28 14:31:51 +00:00
// Leggi il file di hash e cerca l'hash corrispondente
const hashContent = fs.readFileSync(hashFilePath, 'utf8');
const hashLines = hashContent.split('\n');
let expectedHash: string | null = null;
for (const line of hashLines) {
const match = line.match(/^(\w+)\s+(\S+)/);
if (match && match[2] === fileName) {
expectedHash = match[1];
break;
}
}
if (!expectedHash) {
throw new Error('Hash not found for the downloaded file.');
}
// Verifica l'hash del file scaricato
2024-09-29 22:29:16 +00:00
const calculatedHash = await verifyFileHash(`${filePath}/${fileName}`);
2024-09-28 14:31:51 +00:00
return calculatedHash === expectedHash;
};
// Funzione per verificare l'hash del file
const verifyFileHash = (filePath: string): Promise<string> => {
return new Promise((resolve, reject) => {
const hash = createHash('sha256');
const fileStream = fs.createReadStream(filePath);
fileStream.on('data', (data) => {
hash.update(data);
});
fileStream.on('end', () => {
resolve(hash.digest('hex'));
});
fileStream.on('error', (err) => {
reject(err);
});
});
};
2024-10-09 19:44:37 +00:00
const extractTarBz2 = (filePath: string, destination: string): Promise<string> => {
2024-09-29 22:29:16 +00:00
return new Promise((resolve, reject) => {
// Crea il file decomprimendo il .bz2 in uno .tar temporaneo
const tarPath = path.join(destination, 'temp.tar');
const fileStream = fs.createReadStream(filePath);
const decompressedStream = fileStream.pipe(bz2());
const writeStream = fs.createWriteStream(tarPath);
decompressedStream.pipe(writeStream);
2024-10-09 19:44:37 +00:00
let extractedDir: string = '';
2024-09-29 22:29:16 +00:00
writeStream.on('finish', () => {
// Una volta che il file .tar è stato creato, estrailo
2024-10-09 19:44:37 +00:00
tar.extract({ cwd: destination, file: tarPath, onReadEntry: (entry: tar.ReadEntry) => {
if (extractedDir == '') {
const topLevelDir = entry.path.split('/')[0];
extractedDir = topLevelDir; // Salva la prima directory
}
} })
2024-09-29 22:29:16 +00:00
.then(() => {
// Elimina il file .tar temporaneo dopo l'estrazione
fs.unlink(tarPath, (err) => {
if (err) reject(err);
2024-10-09 19:44:37 +00:00
else if (extractedDir == '') reject('Extraction failed')
else resolve(extractedDir);
2024-09-29 22:29:16 +00:00
});
})
.catch(reject);
});
writeStream.on('error', reject);
});
};
2024-10-25 16:17:48 +00:00
const extractZip = (filePath: string, destination: string): Promise<string> => {
return new Promise<string>((resolve, reject) => {
try {
const zip = new AdmZip(filePath);
// Ensure destination exists
if (!fs.existsSync(destination)) {
fs.mkdirSync(destination, { recursive: true });
}
// Extract the ZIP file
zip.extractAllTo(destination, true);
// Get the name of the extracted folder
const extractedEntries = zip.getEntries();
const folderName = extractedEntries[0]?.entryName.split('/')[0];
// Ensure folder name exists
if (!folderName) {
reject(new Error("Could not determine the extracted folder name"));
return;
}
resolve(path.join(destination, folderName));
} catch (error) {
reject(error);
}
});
};
const extract = (filePath: string, destination: string): Promise<string> => {
if (filePath.endsWith('.zip')) {
return extractZip(filePath, destination);
}
else if (filePath.endsWith('.tar.bz2')) {
return extractTarBz2(filePath, destination);
}
throw new Error("Unknown file type " + filePath);
}
2024-10-21 20:36:05 +00:00
// #endregion
function showNotification(options?: NotificationConstructorOptions): void {
2024-10-19 13:57:57 +00:00
if (!options) {
return;
}
if (!options.icon) {
options.icon = wdwIcon;
}
new Notification(options).show();
}
2024-09-17 18:33:24 +00:00
try {
// This method will be called when Electron has finished
// initialization and is ready to create browser windows.
// Some APIs can only be used after this event occurs.
// Added 400 ms to fix the black background issue while using transparent window. More detais at https://github.com/electron/electron/issues/15947
2024-10-29 14:42:58 +00:00
app.on('ready', () => {
Menu.setApplicationMenu(null);
const gotInstanceLock = app.requestSingleInstanceLock();
if (!gotInstanceLock) {
dialog.showErrorBox('', 'Another instance of monerod GUI is running');
app.quit();
return;
}
2024-10-29 17:34:25 +00:00
setTimeout(async () => {
const splash = await createSplashWindow();
createWindow();
await new Promise<void>((resolve, reject) => {
try {
setTimeout(() => {
if (splash) splash.close();
if (!minimized) {
win?.show();
win?.maximize();
}
2024-10-29 17:34:25 +00:00
resolve();
}, 2600);
}
catch(error: any) {
reject(error);
}
});
}, 400);
2024-10-29 14:42:58 +00:00
});
2024-09-17 18:33:24 +00:00
// Quit when all windows are closed.
app.on('window-all-closed', () => {
// On OS X it is common for applications and their menu bar
// to stay active until the user quits explicitly with Cmd + Q
if (process.platform !== 'darwin') {
app.quit();
}
});
app.on('activate', () => {
// On OS X it's common to re-create a window in the app when the
// dock icon is clicked and there are no other windows open.
if (win === null) {
createWindow();
}
});
2024-10-12 12:28:16 +00:00
app.on('before-quit', () => {
isQuitting = true;
});
2024-10-23 15:17:33 +00:00
// #region Security
app.on('web-contents-created', (event, webContents) => {
webContents.setWindowOpenHandler((details) => {
console.warn("Prevented unsafe window creation");
console.warn(details);
return { action: 'deny' };
});
});
app.on('web-contents-created', (event, contents) => {
contents.on('will-navigate', (event, navigationUrl) => {
event.preventDefault();
console.warn(`Prevented unsage window navigation to ${navigationUrl}`);
/*
const parsedUrl = new URL(navigationUrl)
if (parsedUrl.origin !== 'https://example.com') {
event.preventDefault()
}
*/
})
})
// #endregion
2024-10-19 20:07:40 +00:00
2024-10-23 17:01:24 +00:00
ipcMain.handle('is-auto-launched', (event: IpcMainInvokeEvent) => {
console.debug(event);
win?.webContents.send('on-is-auto-launched', isAutoLaunched);
});
ipcMain.handle('quit', (event: IpcMainInvokeEvent) => {
2024-10-13 18:47:25 +00:00
isQuitting = true;
2024-11-07 17:07:31 +00:00
tray.destroy();
win?.close();
win?.destroy();
2024-10-13 18:47:25 +00:00
app.quit();
});
ipcMain.handle('start-monerod', (event: IpcMainInvokeEvent, configFilePath: string[]) => {
2024-09-26 16:45:28 +00:00
startMoneroDaemon(configFilePath);
2024-09-22 13:55:03 +00:00
})
ipcMain.handle('get-monero-version', (event: IpcMainInvokeEvent, configFilePath: string) => {
2024-09-26 16:45:28 +00:00
getMonerodVersion(configFilePath);
});
2024-09-28 14:31:51 +00:00
// Gestione IPC
ipcMain.handle('download-monerod', async (event: IpcMainInvokeEvent, downloadUrl: string, destination: string) => {
2024-09-28 14:31:51 +00:00
try {
2024-09-29 22:29:16 +00:00
//const fileName = path.basename(downloadUrl);
//const filePath = path.join(destination, fileName);
2024-09-28 14:31:51 +00:00
const hashUrl = 'https://www.getmonero.org/downloads/hashes.txt';
// Inizializza il progresso
2024-10-09 18:53:38 +00:00
event.sender.send('download-progress', { progress: 0, status: 'Starting download' });
win?.setProgressBar(0, {
mode: 'normal'
});
2024-09-28 14:31:51 +00:00
// Scarica il file Monero
2024-09-29 22:29:16 +00:00
const fileName = await downloadFile(downloadUrl, destination, (progress) => {
win?.setProgressBar(progress, {
mode: 'normal'
});
2024-10-09 18:53:38 +00:00
event.sender.send('download-progress', { progress, status: 'Downloading' });
2024-09-28 14:31:51 +00:00
});
// Scarica e verifica l'hash
2024-10-09 18:53:38 +00:00
event.sender.send('download-progress', { progress: 100, status: 'Verifying hash' });
win?.setProgressBar(100, {
mode: 'indeterminate'
});
2024-09-29 22:29:16 +00:00
await downloadAndVerifyHash(hashUrl, fileName, destination);
2024-09-28 14:31:51 +00:00
// Estrai il file
2024-10-09 18:53:38 +00:00
const fPath = `${destination}/${fileName}`;
event.sender.send('download-progress', { progress: 100, status: 'Extracting' });
2024-10-25 16:17:48 +00:00
const extractedDir = await extract(fPath, destination);
2024-09-28 14:31:51 +00:00
2024-10-09 18:53:38 +00:00
event.sender.send('download-progress', { progress: 100, status: 'Download and extraction completed successfully' });
2024-11-01 20:45:25 +00:00
event.sender.send('download-progress', { progress: 200, status: os.platform() == 'win32' ? extractedDir : `${destination}/${extractedDir}` });
win?.setProgressBar(100, {
mode: 'none'
});
2024-09-28 14:31:51 +00:00
} catch (error) {
event.sender.send('download-progress', { progress: 0, status: `Error: ${error}` });
2024-11-01 20:45:25 +00:00
win?.setProgressBar(0, {
mode: 'error'
});
2024-09-28 14:31:51 +00:00
}
});
ipcMain.handle('read-file', (event: IpcMainInvokeEvent, filePath: string) => {
fs.readFile(filePath, 'utf-8', (err, data) => {
if (err != null) {
win?.webContents.send('on-read-file-error', `${err}`);
return;
}
win?.webContents.send('on-read-file', data);
});
});
ipcMain.handle('save-file', async (event: IpcMainInvokeEvent, defaultPath: string, content: string) => {
if (!win) {
return;
}
const result = await dialog.showSaveDialog(win, {
title: 'Save File',
defaultPath: defaultPath,
properties: [
'showOverwriteConfirmation'
]
});
if (result.canceled) {
win.webContents.send('on-save-file', '');
return;
}
try {
fs.writeFileSync(result.filePath, content);
win.webContents.send('on-save-file', result.filePath);
}
catch(error: any) {
win.webContents.send('on-save-file-error', `${error}`);
}
});
ipcMain.handle('select-file', async (event: IpcMainInvokeEvent, extensions?: string[]) => {
if (!win)
{
return;
}
const result = await dialog.showOpenDialog(win, {
title: 'Select File',
2024-10-19 10:27:40 +00:00
filters: extensions ? [{
name: 'filter',
extensions: extensions
}] : [],
properties: ['openFile']
});
const path = result.canceled ? null : result.filePaths[0];
win.webContents.send('selected-file', path ? `${path}` : '');
});
2024-09-28 14:31:51 +00:00
ipcMain.handle('select-folder', async (event: IpcMainInvokeEvent) => {
if (!win) {
return;
}
const result = await dialog.showOpenDialog(win, {
title: 'Select Folder',
2024-10-09 18:53:38 +00:00
properties: ['openDirectory'], // Specifica che vogliamo solo cartelle
});
const path = result.canceled ? null : result.filePaths[0];
win.webContents.send('selected-folder', path ? `${path}` : '');
2024-10-09 18:53:38 +00:00
});
2024-09-28 14:31:51 +00:00
ipcMain.handle('get-path', (event: IpcMainInvokeEvent, path: 'home' | 'appData' | 'userData' | 'sessionData' | 'temp' | 'exe' | 'module' | 'desktop' | 'documents' | 'downloads' | 'music' | 'pictures' | 'videos' | 'recent' | 'logs' | 'crashDumps') => {
win?.webContents.send('on-get-path', app.getPath(path));
});
ipcMain.handle('is-wifi-connected', async (event: IpcMainInvokeEvent) => {
isWifiConnected();
2024-10-09 21:40:32 +00:00
});
ipcMain.handle('get-os-type', (event: IpcMainInvokeEvent) => {
2024-10-09 21:40:32 +00:00
win?.webContents.send('got-os-type', { platform: os.platform(), arch: os.arch() });
})
2024-10-16 16:18:43 +00:00
ipcMain.handle('monitor-monerod', (event: IpcMainInvokeEvent) => {
monitorMonerod();
});
ipcMain.handle('check-valid-monerod-path', (event: IpcMainInvokeEvent, path: string) => {
checkValidMonerodPath(path);
})
ipcMain.handle('show-notification', (event: IpcMainInvokeEvent, options?: NotificationConstructorOptions) => {
showNotification(options);
});
// #region Auto Launch
2024-10-21 20:36:05 +00:00
ipcMain.handle('is-auto-launch-enabled', (event: IpcMainInvokeEvent) => {
autoLauncher.isEnabled().then((enabled: boolean) => {
win?.webContents.send('on-is-auto-launch-enabled', enabled);
}).catch((error: any) => {
console.error(error);
win?.webContents.send('on-is-auto-launch-enabled', false);
});
});
ipcMain.handle('enable-auto-launch', (event: IpcMainInvokeEvent, minimized: boolean) => {
2024-10-21 20:36:05 +00:00
autoLauncher.isEnabled().then((enabled: boolean) => {
if (enabled) {
win?.webContents.send('on-enable-auto-launch-error', 'already enabled');
return;
}
autoLauncher = new AutoLaunch({
name: 'monerod-gui',
path: process.execPath,
options: {
launchInBackground: minimized,
extraArguments: [
'--auto-launch'
]
}
});
2024-10-21 20:36:05 +00:00
autoLauncher.enable().then(() => {
autoLauncher.isEnabled().then((enabled: boolean) => {
if (enabled) {
win?.webContents.send('on-enable-auto-launch-success');
}
win?.webContents.send('on-enable-auto-launch-error', `Could not enabled auto launch`);
}).catch((error: any) => {
win?.webContents.send('on-enable-auto-launch-error', `${error}`);
});
}).catch((error: any) => {
console.error(error);
win?.webContents.send('on-enable-auto-launch-error', `${error}`);
});
}).catch((error: any) => {
console.error(error);
win?.webContents.send('on-enable-auto-launch-error', `${error}`);
});
});
2024-11-08 17:36:05 +00:00
ipcMain.handle('get-battery-level', (event: IpcMainInvokeEvent) => {
batteryLevel().then((level: number) => {
win?.webContents.send('on-get-battery-level', level);
}).catch((error: any) => {
console.error(error);
win?.webContents.send('on-get-battery-level', -1);
})
});
ipcMain.handle('disable-auto-launch', (event: IpcMainInvokeEvent) => {
2024-10-21 20:36:05 +00:00
autoLauncher.isEnabled().then((enabled: boolean) => {
if (!enabled) {
win?.webContents.send('on-disable-auto-launch-error', 'already disabled');
return;
}
autoLauncher.disable().then(() => {
autoLauncher.isEnabled().then((enabled: boolean) => {
if (!enabled) {
win?.webContents.send('on-disable-auto-launch-success');
}
win?.webContents.send('on-disable-auto-launch-error', `Could not disable auto launch`);
}).catch((error: any) => {
win?.webContents.send('on-disable-auto-launch-error', `${error}`);
});
}).catch((error: any) => {
console.error(error);
win?.webContents.send('on-disable-auto-launch-error', `${error}`);
});
}).catch((error: any) => {
console.error(error);
win?.webContents.send('on-disable-auto-launch-error', `${error}`);
});
});
// #endregion
2024-11-01 13:39:47 +00:00
ipcMain.handle('set-tray-item-enabled', (event: IpcMainInvokeEvent, id: string, enabled: boolean) => {
setTrayItemEnabled(id, enabled);
});
ipcMain.handle('set-tray-tool-tip', (event: IpcMainInvokeEvent, toolTip: string) => {
tray.setToolTip(toolTip);
});
2024-10-21 20:36:05 +00:00
ipcMain.handle('is-app-image', (event: IpcMainInvokeEvent) => {
win?.webContents.send('on-is-app-image', isAppImage());
2024-10-21 20:36:05 +00:00
});
ipcMain.handle('copy-to-clipboard', (event: IpcMainInvokeEvent, content: string) => {
clipboard.writeText(content, "selection");
});
2024-09-17 18:33:24 +00:00
} catch (e) {
// Catch Error
2024-10-16 16:18:43 +00:00
console.error(e);
2024-09-17 18:33:24 +00:00
// throw e;
}