mirror of
https://github.com/Retropex/umbrel-bitcoin.git
synced 2025-05-12 11:10:48 +02:00
166 lines
4.8 KiB
JavaScript
166 lines
4.8 KiB
JavaScript
#!/usr/bin/env node
|
|
|
|
/**
|
|
* Module dependencies.
|
|
*/
|
|
const fs = require('fs');
|
|
var app = require('../app');
|
|
var debug = require('debug')('nodejs-regular-webapp2:server');
|
|
var http = require('http');
|
|
|
|
const configLogic = require('../logic/config');
|
|
const diskService = require('../services/disk'); // CHECK THIS
|
|
const bitcoindLogic = require('../logic/bitcoind');
|
|
const constants = require('../utils/const');
|
|
|
|
// IIFE to check if umbrel-bitcoin.conf is out-of-date (e.g., due to new config options introduced by an app update)
|
|
(async () => {
|
|
const config = await configLogic.getJsonStore();
|
|
|
|
if(! await configLogic.isUmbrelBitcoinConfUpToDate(config)) {
|
|
console.log('umbrel-bitcoin.conf is not up to date, generating new config...');
|
|
|
|
// set torProxyForClearnet to false for pre-advanced-settings installs that are just now getting advanced settings
|
|
if (constants.BITCOIN_INITIALIZE_WITH_CLEARNET_OVER_TOR) config.torProxyForClearnet = true;
|
|
|
|
await configLogic.applyCustomBitcoinConfig(config);
|
|
await bitcoindLogic.restartBitcoindWithRetries();
|
|
}
|
|
})();
|
|
|
|
// IIFE to check if settings.json is invalid
|
|
(async () => {
|
|
// Check and fix invalid JSON in settings.json
|
|
const JSON_VALIDATION_INTERVAL_MS = 60000; // 1 minute
|
|
|
|
setInterval(async () => {
|
|
// Fetch the current network/chain from user's settings
|
|
// This can change after server initialization if the user changes the network/chain in the UI
|
|
const config = await configLogic.getJsonStore();
|
|
const network = config.network;
|
|
|
|
const SETTINGS_DIR = getSettingsDir(network);
|
|
const SETTINGS_PATH = `${SETTINGS_DIR}/settings.json`;
|
|
|
|
try {
|
|
await diskService.readJsonFile(SETTINGS_PATH);
|
|
} catch (error) {
|
|
if (error instanceof SyntaxError) {
|
|
console.log(`[WARNING] Invalid JSON detected in ${SETTINGS_PATH}. Overwriting with empty JSON.`);
|
|
// We create a backup of the settings.json file before overwriting it with empty JSON. Creating a backup is likely unecessary, because this file is primarily used for bitcoin-qt settings, which are not used by Umbrel. However, it is better to be safe than sorry.
|
|
await createBackup(SETTINGS_PATH);
|
|
await diskService.writeJsonFile(SETTINGS_PATH, {});
|
|
// We do not need to restart bitcoind here because:
|
|
// 1. At a 1 minute check-interval, we likely fix the issue before bitcoind notices if an issue occurs while bitcoind is running
|
|
// 2. if bitcoind errors during operation or during startup (say from a corrupted json file due to bad shutdown), the container will continually attempt restart until settings.json is fixed
|
|
} else {
|
|
console.error(error);
|
|
}
|
|
}
|
|
}, JSON_VALIDATION_INTERVAL_MS);
|
|
})();
|
|
|
|
/**
|
|
* Get port from environment and store in Express.
|
|
*/
|
|
|
|
var port = normalizePort(process.env.PORT || '3005');
|
|
app.set('port', port);
|
|
|
|
/**
|
|
* Create HTTP server.
|
|
*/
|
|
|
|
var server = http.createServer(app);
|
|
|
|
/**
|
|
* Listen on provided port, on all network interfaces.
|
|
*/
|
|
|
|
server.listen(port);
|
|
server.on('error', onError);
|
|
server.on('listening', onListening);
|
|
|
|
/**
|
|
* Normalize a port into a number, string, or false.
|
|
*/
|
|
|
|
function normalizePort(val) {
|
|
var port = parseInt(val, 10);
|
|
|
|
if (isNaN(port)) {
|
|
// named pipe
|
|
return val;
|
|
}
|
|
|
|
if (port >= 0) {
|
|
// port number
|
|
return port;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Event listener for HTTP server "error" event.
|
|
*/
|
|
|
|
function onError(error) {
|
|
if (error.syscall !== 'listen') {
|
|
throw error;
|
|
}
|
|
|
|
var bind = typeof port === 'string'
|
|
? 'Pipe ' + port
|
|
: 'Port ' + port;
|
|
|
|
// handle specific listen errors with friendly messages
|
|
switch (error.code) {
|
|
case 'EACCES':
|
|
console.error(bind + ' requires elevated privileges');
|
|
process.exit(1);
|
|
break;
|
|
case 'EADDRINUSE':
|
|
console.error(bind + ' is already in use');
|
|
process.exit(1);
|
|
break;
|
|
default:
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Event listener for HTTP server "listening" event.
|
|
*/
|
|
|
|
async function onListening() {
|
|
var addr = server.address();
|
|
var bind = typeof addr === 'string'
|
|
? 'pipe ' + addr
|
|
: 'port ' + addr.port;
|
|
debug('Listening on ' + bind);
|
|
console.log('Listening on ' + bind);
|
|
}
|
|
|
|
// Determines the settings.json directory based on the network
|
|
function getSettingsDir(network) {
|
|
if (network === 'main') {
|
|
return '/bitcoin/.bitcoin';
|
|
} else if (network === 'test') {
|
|
return '/bitcoin/.bitcoin/testnet3';
|
|
} else {
|
|
return `/bitcoin/.bitcoin/${network}`;
|
|
}
|
|
}
|
|
|
|
// Creates a backup of the settings.json file
|
|
async function createBackup(settingsPath) {
|
|
try {
|
|
const timestamp = Date.now();
|
|
const backupPath = `${settingsPath}.backup.${timestamp}`;
|
|
await fs.promises.copyFile(settingsPath, backupPath);
|
|
console.log(`settings.json backup created at ${backupPath}`);
|
|
} catch (error) {
|
|
console.error(`Failed to create backup: ${error}`);
|
|
}
|
|
} |