Browse Source
- Completely rewrite main.js for reliable asset loading - Update preload.js with proper security context isolation - Fix file:// protocol handling for application resources - Add proper error logging and reporting in Electron context - Disable service workers in Electron environment - Fix path resolution for assets in packaged application - Add prerequisite checking for Electron builds - Update electron-builder configuration The changes resolve persistent issues with: 1. Missing assets in packaged application 2. Incorrect path resolution in production builds 3. Service worker conflicts in desktop environment 4. Security context handling in preload script 5. Electron build process reliabilityelectron_fix_20250317
13 changed files with 694 additions and 271 deletions
@ -0,0 +1,36 @@ |
|||
{ |
|||
"appId": "app.timesafari.app", |
|||
"productName": "TimeSafari", |
|||
"directories": { |
|||
"output": "dist-electron-packages", |
|||
"buildResources": "build" |
|||
}, |
|||
"files": [ |
|||
"dist-electron/**/*", |
|||
"node_modules/**/*", |
|||
"package.json", |
|||
"src/electron/electron-logger.js" |
|||
], |
|||
"extraResources": [ |
|||
{ |
|||
"from": "src/utils", |
|||
"to": "utils", |
|||
"filter": ["**/*"] |
|||
} |
|||
], |
|||
"extraMetadata": { |
|||
"main": "src/electron/main.js" |
|||
}, |
|||
"linux": { |
|||
"target": ["AppImage"], |
|||
"category": "Utility", |
|||
"maintainer": "TimeSafari Team" |
|||
}, |
|||
"mac": { |
|||
"target": ["dmg"], |
|||
"category": "public.app-category.productivity" |
|||
}, |
|||
"win": { |
|||
"target": ["nsis"] |
|||
} |
|||
} |
@ -0,0 +1,177 @@ |
|||
#!/usr/bin/env node
|
|||
|
|||
/** |
|||
* @file check-electron-prerequisites.js |
|||
* @description Verifies and installs required dependencies for Electron builds |
|||
* |
|||
* This script checks if Python's distutils module is available, which is required |
|||
* by node-gyp when compiling native Node.js modules during Electron packaging. |
|||
* Without distutils, builds will fail with "ModuleNotFoundError: No module named 'distutils'". |
|||
* |
|||
* The script performs the following actions: |
|||
* 1. Checks if Python's distutils module is available |
|||
* 2. If missing, offers to install setuptools package which provides distutils |
|||
* 3. Attempts installation through pip or pip3 |
|||
* 4. Provides manual installation instructions if automated installation fails |
|||
* |
|||
* Usage: |
|||
* - Direct execution: node scripts/check-electron-prerequisites.js |
|||
* - As npm script: npm run check:electron |
|||
* - Before builds: npm run check:electron && electron-builder |
|||
* |
|||
* Exit codes: |
|||
* - 0: All prerequisites are met or were successfully installed |
|||
* - 1: Prerequisites are missing and weren't installed |
|||
* |
|||
* @author [YOUR_NAME] |
|||
* @version 1.0.0 |
|||
* @license MIT |
|||
*/ |
|||
|
|||
const { execSync } = require('child_process'); |
|||
const readline = require('readline'); |
|||
const chalk = require('chalk'); // You might need to add this to your dependencies
|
|||
|
|||
console.log(chalk.blue('🔍 Checking Electron build prerequisites...')); |
|||
|
|||
/** |
|||
* Checks if Python's distutils module is available |
|||
* |
|||
* This function attempts to import the distutils module in Python. |
|||
* If successful, it means node-gyp will be able to compile native modules. |
|||
* If unsuccessful, the Electron build will likely fail when compiling native dependencies. |
|||
* |
|||
* @returns {boolean} True if distutils is available, false otherwise |
|||
* |
|||
* @example |
|||
* if (checkDistutils()) { |
|||
* console.log('Ready to build Electron app'); |
|||
* } |
|||
*/ |
|||
function checkDistutils() { |
|||
try { |
|||
// Attempt to import distutils using Python
|
|||
// We use stdio: 'ignore' to suppress any Python output
|
|||
execSync('python -c "import distutils"', { stdio: 'ignore' }); |
|||
console.log(chalk.green('✅ Python distutils is available')); |
|||
return true; |
|||
} catch (e) { |
|||
// This error occurs if either Python is not found or if distutils is missing
|
|||
console.log(chalk.red('❌ Python distutils is missing')); |
|||
return false; |
|||
} |
|||
} |
|||
|
|||
/** |
|||
* Installs the setuptools package which provides distutils |
|||
* |
|||
* This function attempts to install setuptools using pip or pip3. |
|||
* Setuptools is a package that provides the distutils module needed by node-gyp. |
|||
* In Python 3.12+, distutils was moved out of the standard library into setuptools. |
|||
* |
|||
* The function tries multiple installation methods: |
|||
* 1. First attempts with pip |
|||
* 2. If that fails, tries with pip3 |
|||
* 3. If both fail, provides instructions for manual installation |
|||
* |
|||
* @returns {Promise<boolean>} True if installation succeeded, false otherwise |
|||
* |
|||
* @example |
|||
* const success = await installSetuptools(); |
|||
* if (success) { |
|||
* console.log('Ready to proceed with build'); |
|||
* } else { |
|||
* console.log('Please fix prerequisites manually'); |
|||
* } |
|||
*/ |
|||
async function installSetuptools() { |
|||
console.log(chalk.yellow('📦 Attempting to install setuptools...')); |
|||
|
|||
try { |
|||
// First try with pip, commonly used on all platforms
|
|||
execSync('pip install setuptools', { stdio: 'inherit' }); |
|||
console.log(chalk.green('✅ Successfully installed setuptools')); |
|||
return true; |
|||
} catch (pipError) { |
|||
try { |
|||
// If pip fails, try with pip3 (common on Linux distributions)
|
|||
console.log(chalk.yellow('⚠️ Trying with pip3...')); |
|||
execSync('pip3 install setuptools', { stdio: 'inherit' }); |
|||
console.log(chalk.green('✅ Successfully installed setuptools using pip3')); |
|||
return true; |
|||
} catch (pip3Error) { |
|||
// If both methods fail, provide manual installation guidance
|
|||
console.log(chalk.red('❌ Failed to install setuptools automatically')); |
|||
console.log(chalk.yellow('📝 Please install it manually with:')); |
|||
console.log(' pip install setuptools'); |
|||
console.log(' or'); |
|||
console.log(' sudo apt install python3-setuptools (on Debian/Ubuntu)'); |
|||
console.log(' sudo pacman -S python-setuptools (on Arch Linux)'); |
|||
console.log(' sudo dnf install python3-setuptools (on Fedora)'); |
|||
console.log(' brew install python-setuptools (on macOS with Homebrew)'); |
|||
return false; |
|||
} |
|||
} |
|||
} |
|||
|
|||
/** |
|||
* Main execution function |
|||
* |
|||
* This function orchestrates the checking and installation process: |
|||
* 1. Checks if distutils is already available |
|||
* 2. If not, informs the user and prompts for installation |
|||
* 3. Based on user input, attempts to install or exits |
|||
* |
|||
* The function handles interactive user prompts and orchestrates |
|||
* the overall flow of the script. |
|||
* |
|||
* @returns {Promise<void>} |
|||
* @throws Will exit process with code 1 if prerequisites aren't met |
|||
*/ |
|||
async function main() { |
|||
// First check if distutils is already available
|
|||
if (checkDistutils()) { |
|||
// All prerequisites are met, exit successfully
|
|||
process.exit(0); |
|||
} |
|||
|
|||
// Inform the user about the missing prerequisite
|
|||
console.log(chalk.yellow('⚠️ Python distutils is required for Electron builds')); |
|||
console.log(chalk.yellow('⚠️ This is needed to compile native modules during the build process')); |
|||
|
|||
// Set up readline interface for user interaction
|
|||
const rl = readline.createInterface({ |
|||
input: process.stdin, |
|||
output: process.stdout |
|||
}); |
|||
|
|||
// Prompt the user for installation permission
|
|||
const answer = await new Promise(resolve => { |
|||
rl.question(chalk.blue('Would you like to install setuptools now? (y/n) '), resolve); |
|||
}); |
|||
|
|||
// Clean up readline interface
|
|||
rl.close(); |
|||
|
|||
if (answer.toLowerCase() === 'y') { |
|||
// User agreed to installation
|
|||
const success = await installSetuptools(); |
|||
if (success) { |
|||
// Installation succeeded, exit successfully
|
|||
process.exit(0); |
|||
} else { |
|||
// Installation failed, exit with error
|
|||
process.exit(1); |
|||
} |
|||
} else { |
|||
// User declined installation
|
|||
console.log(chalk.yellow('⚠️ Build may fail without distutils')); |
|||
process.exit(1); |
|||
} |
|||
} |
|||
|
|||
// Execute the main function and handle any uncaught errors
|
|||
main().catch(error => { |
|||
console.error(chalk.red('Error during prerequisites check:'), error); |
|||
process.exit(1); |
|||
}); |
@ -0,0 +1,61 @@ |
|||
/** |
|||
* Fix path resolution issues in the Electron build |
|||
*/ |
|||
const fs = require('fs'); |
|||
const path = require('path'); |
|||
const glob = require('glob'); |
|||
|
|||
// Fix asset paths in HTML file
|
|||
function fixHtmlPaths() { |
|||
const htmlFile = path.join(__dirname, '../dist-electron/index.html'); |
|||
if (fs.existsSync(htmlFile)) { |
|||
let html = fs.readFileSync(htmlFile, 'utf8'); |
|||
|
|||
// Convert absolute paths to relative
|
|||
html = html.replace(/src="\//g, 'src="./'); |
|||
html = html.replace(/href="\//g, 'href="./'); |
|||
|
|||
fs.writeFileSync(htmlFile, html); |
|||
console.log('✅ Fixed paths in index.html'); |
|||
} |
|||
} |
|||
|
|||
// Fix asset imports in JS files
|
|||
function fixJsPaths() { |
|||
const jsFiles = glob.sync('dist-electron/assets/*.js'); |
|||
|
|||
jsFiles.forEach(file => { |
|||
let content = fs.readFileSync(file, 'utf8'); |
|||
|
|||
// Replace absolute imports with relative ones
|
|||
const originalContent = content; |
|||
content = content.replace(/["']\/assets\//g, '"./assets/'); |
|||
|
|||
if (content !== originalContent) { |
|||
fs.writeFileSync(file, content); |
|||
console.log(`✅ Fixed paths in ${path.basename(file)}`); |
|||
} |
|||
}); |
|||
} |
|||
|
|||
// Add base href to HTML
|
|||
function addBaseHref() { |
|||
const htmlFile = path.join(__dirname, '../dist-electron/index.html'); |
|||
if (fs.existsSync(htmlFile)) { |
|||
let html = fs.readFileSync(htmlFile, 'utf8'); |
|||
|
|||
// Add base href if not present
|
|||
if (!html.includes('<base href=')) { |
|||
html = html.replace('</head>', '<base href="./">\n</head>'); |
|||
fs.writeFileSync(htmlFile, html); |
|||
console.log('✅ Added base href to index.html'); |
|||
} |
|||
} |
|||
} |
|||
|
|||
// Run all fixes
|
|||
fixHtmlPaths(); |
|||
fixJsPaths(); |
|||
addBaseHref(); |
|||
|
|||
console.log('🎉 Electron path fixes completed'); |
@ -0,0 +1,14 @@ |
|||
// This is a placeholder notarize script that does nothing for non-macOS platforms
|
|||
// Only necessary for macOS app store submissions
|
|||
|
|||
exports.default = async function notarizing(context) { |
|||
// Only notarize macOS builds
|
|||
if (context.electronPlatformName !== 'darwin') { |
|||
console.log('Skipping notarization for non-macOS platform'); |
|||
return; |
|||
} |
|||
|
|||
// For macOS, we would implement actual notarization here
|
|||
console.log('This is where macOS notarization would happen'); |
|||
// We're just returning with no action for non-macOS builds
|
|||
}; |
@ -0,0 +1,33 @@ |
|||
/** |
|||
* Electron-specific logger implementation |
|||
*/ |
|||
const fs = require('fs'); |
|||
const path = require('path'); |
|||
const { app } = require('electron'); |
|||
|
|||
// Create logs directory if it doesn't exist
|
|||
const logsDir = path.join(app.getPath('userData'), 'logs'); |
|||
if (!fs.existsSync(logsDir)) { |
|||
fs.mkdirSync(logsDir, { recursive: true }); |
|||
} |
|||
|
|||
const logFile = path.join(logsDir, `electron-${new Date().toISOString().split('T')[0]}.log`); |
|||
|
|||
function log(level, message) { |
|||
const timestamp = new Date().toISOString(); |
|||
const logMessage = `[${timestamp}] [${level}] ${message}\n`; |
|||
|
|||
// Write to log file
|
|||
fs.appendFileSync(logFile, logMessage); |
|||
|
|||
// Also output to console
|
|||
console[level.toLowerCase()](message); |
|||
} |
|||
|
|||
module.exports = { |
|||
info: (message) => log('INFO', message), |
|||
warn: (message) => log('WARN', message), |
|||
error: (message) => log('ERROR', message), |
|||
debug: (message) => log('DEBUG', message), |
|||
getLogPath: () => logFile |
|||
}; |
@ -1,174 +1,217 @@ |
|||
const { app, BrowserWindow } = require("electron"); |
|||
const path = require("path"); |
|||
const fs = require("fs"); |
|||
const logger = require("../utils/logger"); |
|||
const { app, BrowserWindow, session, protocol, ipcMain, dialog } = require('electron'); |
|||
const path = require('path'); |
|||
const fs = require('fs'); |
|||
const url = require('url'); |
|||
|
|||
// Check if running in dev mode
|
|||
const isDev = process.argv.includes("--inspect"); |
|||
// Global window reference
|
|||
let mainWindow = null; |
|||
|
|||
function createWindow() { |
|||
// Add before createWindow function
|
|||
const preloadPath = path.join(__dirname, "preload.js"); |
|||
logger.log("Checking preload path:", preloadPath); |
|||
logger.log("Preload exists:", fs.existsSync(preloadPath)); |
|||
// Debug flags
|
|||
const isDev = !app.isPackaged; |
|||
|
|||
// Helper for logging
|
|||
function logDebug(...args) { |
|||
console.log('[DEBUG]', ...args); |
|||
} |
|||
|
|||
function logError(...args) { |
|||
console.error('[ERROR]', ...args); |
|||
if (!isDev && mainWindow) { |
|||
dialog.showErrorBox('TimeSafari Error', args.join(' ')); |
|||
} |
|||
} |
|||
|
|||
// Create the browser window.
|
|||
const mainWindow = new BrowserWindow({ |
|||
// Get the most appropriate app path
|
|||
function getAppPath() { |
|||
if (app.isPackaged) { |
|||
const possiblePaths = [ |
|||
path.join(process.resourcesPath, 'app.asar', 'dist-electron'), |
|||
path.join(process.resourcesPath, 'app.asar'), |
|||
path.join(process.resourcesPath, 'app'), |
|||
app.getAppPath() |
|||
]; |
|||
|
|||
for (const testPath of possiblePaths) { |
|||
const testFile = path.join(testPath, 'www', 'index.html'); |
|||
if (fs.existsSync(testFile)) { |
|||
logDebug(`Found valid app path: ${testPath}`); |
|||
return testPath; |
|||
} |
|||
} |
|||
|
|||
logError('Could not find valid app path'); |
|||
return path.join(process.resourcesPath, 'app.asar'); // Default fallback
|
|||
} else { |
|||
return __dirname; |
|||
} |
|||
} |
|||
|
|||
// Create the browser window
|
|||
function createWindow() { |
|||
logDebug('Creating window with paths:'); |
|||
logDebug('- process.resourcesPath:', process.resourcesPath); |
|||
logDebug('- app.getAppPath():', app.getAppPath()); |
|||
logDebug('- __dirname:', __dirname); |
|||
|
|||
// Create the browser window
|
|||
mainWindow = new BrowserWindow({ |
|||
width: 1200, |
|||
height: 800, |
|||
webPreferences: { |
|||
nodeIntegration: false, |
|||
preload: path.join(__dirname, 'preload.js'), |
|||
contextIsolation: true, |
|||
webSecurity: true, |
|||
allowRunningInsecureContent: false, |
|||
preload: path.join(__dirname, "preload.js"), |
|||
}, |
|||
nodeIntegration: false, |
|||
webSecurity: true |
|||
} |
|||
}); |
|||
|
|||
// Always open DevTools for now
|
|||
mainWindow.webContents.openDevTools(); |
|||
|
|||
// Intercept requests to fix asset paths
|
|||
mainWindow.webContents.session.webRequest.onBeforeRequest( |
|||
{ |
|||
urls: [ |
|||
"file://*/*/assets/*", |
|||
"file://*/assets/*", |
|||
"file:///assets/*", // Catch absolute paths
|
|||
"<all_urls>", // Catch all URLs as a fallback
|
|||
], |
|||
}, |
|||
(details, callback) => { |
|||
let url = details.url; |
|||
|
|||
// Handle paths that don't start with file://
|
|||
if (!url.startsWith("file://") && url.includes("/assets/")) { |
|||
url = `file://${path.join(__dirname, "www", url)}`; |
|||
|
|||
// Fix root file paths - replaces all protocol handling
|
|||
protocol.interceptFileProtocol('file', (request, callback) => { |
|||
let urlPath = request.url.substr(7); // Remove 'file://' prefix
|
|||
urlPath = decodeURIComponent(urlPath); // Handle special characters
|
|||
|
|||
// Debug all asset requests
|
|||
if (urlPath.includes('assets/') || urlPath.endsWith('.js') || urlPath.endsWith('.css') || urlPath.endsWith('.html')) { |
|||
logDebug(`Intercepted request for: ${urlPath}`); |
|||
} |
|||
|
|||
// Fix paths for files at root like registerSW.js or manifest.webmanifest
|
|||
if (urlPath.endsWith('registerSW.js') || |
|||
urlPath.endsWith('manifest.webmanifest') || |
|||
urlPath.endsWith('sw.js')) { |
|||
|
|||
const appBasePath = getAppPath(); |
|||
const filePath = path.join(appBasePath, 'www', path.basename(urlPath)); |
|||
|
|||
if (fs.existsSync(filePath)) { |
|||
logDebug(`Serving ${urlPath} from ${filePath}`); |
|||
return callback({ path: filePath }); |
|||
} else { |
|||
// For service worker, provide empty content to avoid errors
|
|||
if (urlPath.endsWith('registerSW.js') || urlPath.endsWith('sw.js')) { |
|||
logDebug(`Providing empty SW file for ${urlPath}`); |
|||
// Create an empty JS file content that does nothing
|
|||
const tempFile = path.join(app.getPath('temp'), path.basename(urlPath)); |
|||
fs.writeFileSync(tempFile, '// Service workers disabled in Electron\n'); |
|||
return callback({ path: tempFile }); |
|||
} |
|||
} |
|||
|
|||
// Handle absolute paths starting with /assets/
|
|||
if (url.includes("/assets/") && !url.includes("/www/assets/")) { |
|||
const baseDir = url.includes("dist-electron") |
|||
? url.substring( |
|||
0, |
|||
url.indexOf("/dist-electron") + "/dist-electron".length, |
|||
) |
|||
: `file://${__dirname}`; |
|||
const assetPath = url.split("/assets/")[1]; |
|||
const newUrl = `${baseDir}/www/assets/${assetPath}`; |
|||
callback({ redirectURL: newUrl }); |
|||
return; |
|||
} |
|||
|
|||
// Handle assets paths that might be requested from root
|
|||
if (urlPath.startsWith('/assets/') || urlPath === '/assets') { |
|||
const appBasePath = getAppPath(); |
|||
const filePath = path.join(appBasePath, 'www', urlPath); |
|||
logDebug(`Redirecting ${urlPath} to ${filePath}`); |
|||
return callback({ path: filePath }); |
|||
} |
|||
|
|||
// Handle assets paths that are missing the www folder
|
|||
if (urlPath.includes('/assets/')) { |
|||
const appBasePath = getAppPath(); |
|||
const relativePath = urlPath.substring(urlPath.indexOf('/assets/')); |
|||
const filePath = path.join(appBasePath, 'www', relativePath); |
|||
if (fs.existsSync(filePath)) { |
|||
logDebug(`Fixing asset path ${urlPath} to ${filePath}`); |
|||
return callback({ path: filePath }); |
|||
} |
|||
|
|||
callback({}); // No redirect for other URLs
|
|||
}, |
|||
); |
|||
|
|||
if (isDev) { |
|||
// Debug info
|
|||
logger.log("Debug Info:"); |
|||
logger.log("Running in dev mode:", isDev); |
|||
logger.log("App is packaged:", app.isPackaged); |
|||
logger.log("Process resource path:", process.resourcesPath); |
|||
logger.log("App path:", app.getAppPath()); |
|||
logger.log("__dirname:", __dirname); |
|||
logger.log("process.cwd():", process.cwd()); |
|||
} |
|||
|
|||
const indexPath = path.join(__dirname, "www", "index.html"); |
|||
|
|||
if (isDev) { |
|||
logger.log("Loading index from:", indexPath); |
|||
logger.log("www path:", path.join(__dirname, "www")); |
|||
logger.log("www assets path:", path.join(__dirname, "www", "assets")); |
|||
} |
|||
|
|||
if (!fs.existsSync(indexPath)) { |
|||
logger.error(`Index file not found at: ${indexPath}`); |
|||
throw new Error("Index file not found"); |
|||
} |
|||
|
|||
// Add CSP headers to allow API connections
|
|||
mainWindow.webContents.session.webRequest.onHeadersReceived( |
|||
(details, callback) => { |
|||
callback({ |
|||
responseHeaders: { |
|||
...details.responseHeaders, |
|||
"Content-Security-Policy": [ |
|||
"default-src 'self';" + |
|||
"connect-src 'self' https://api.endorser.ch https://*.timesafari.app;" + |
|||
"img-src 'self' data: https: blob:;" + |
|||
"script-src 'self' 'unsafe-inline' 'unsafe-eval';" + |
|||
"style-src 'self' 'unsafe-inline';" + |
|||
"font-src 'self' data:;", |
|||
], |
|||
}, |
|||
}); |
|||
}, |
|||
); |
|||
|
|||
// Load the index.html
|
|||
mainWindow |
|||
.loadFile(indexPath) |
|||
.then(() => { |
|||
logger.log("Successfully loaded index.html"); |
|||
if (isDev) { |
|||
mainWindow.webContents.openDevTools(); |
|||
logger.log("DevTools opened - running in dev mode"); |
|||
} |
|||
|
|||
// For all other paths, just pass them through
|
|||
callback({ path: urlPath }); |
|||
}); |
|||
|
|||
// Set up CSP headers - more permissive in dev mode
|
|||
session.defaultSession.webRequest.onHeadersReceived((details, callback) => { |
|||
callback({ |
|||
responseHeaders: { |
|||
...details.responseHeaders, |
|||
'Content-Security-Policy': [ |
|||
isDev |
|||
? "default-src 'self' file:; script-src 'self' 'unsafe-inline' 'unsafe-eval'; style-src 'self' 'unsafe-inline' https://fonts.googleapis.com; font-src 'self' https://fonts.gstatic.com; img-src 'self' data: blob: https://*; connect-src 'self' https://*" |
|||
: "default-src 'self' file:; script-src 'self' 'unsafe-inline' 'unsafe-eval'; style-src 'self' 'unsafe-inline' https://fonts.googleapis.com; font-src 'self' https://fonts.gstatic.com; img-src 'self' data: blob: https://image.timesafari.app https://*.americancloud.com; connect-src 'self' https://api.timesafari.app https://api.endorser.ch https://test-api.endorser.ch https://fonts.googleapis.com" |
|||
] |
|||
} |
|||
}) |
|||
.catch((err) => { |
|||
logger.error("Failed to load index.html:", err); |
|||
logger.error("Attempted path:", indexPath); |
|||
}); |
|||
|
|||
// Listen for console messages from the renderer
|
|||
mainWindow.webContents.on("console-message", (_event, level, message) => { |
|||
logger.log("Renderer Console:", message); |
|||
}); |
|||
|
|||
// Add right after creating the BrowserWindow
|
|||
mainWindow.webContents.on( |
|||
"did-fail-load", |
|||
(event, errorCode, errorDescription) => { |
|||
logger.error("Page failed to load:", errorCode, errorDescription); |
|||
}, |
|||
); |
|||
|
|||
mainWindow.webContents.on("preload-error", (event, preloadPath, error) => { |
|||
logger.error("Preload script error:", preloadPath, error); |
|||
}); |
|||
|
|||
mainWindow.webContents.on( |
|||
"console-message", |
|||
(event, level, message, line, sourceId) => { |
|||
logger.log("Renderer Console:", line, sourceId, message); |
|||
}, |
|||
); |
|||
|
|||
// Enable remote debugging when in dev mode
|
|||
|
|||
// Load the index.html with modifications
|
|||
try { |
|||
const appPath = getAppPath(); |
|||
const wwwFolder = path.join(appPath, 'www'); |
|||
const indexPath = path.join(wwwFolder, 'index.html'); |
|||
|
|||
logDebug('Loading app from:', indexPath); |
|||
|
|||
// Check if the file exists
|
|||
if (fs.existsSync(indexPath)) { |
|||
// Read and modify index.html to disable service worker
|
|||
let indexContent = fs.readFileSync(indexPath, 'utf8'); |
|||
|
|||
// 1. Add base tag for proper path resolution
|
|||
indexContent = indexContent.replace('<head>', |
|||
`<head>\n <base href="file://${wwwFolder}/">`); |
|||
|
|||
// 2. Disable service worker registration by replacing the script
|
|||
if (indexContent.includes('registerSW.js')) { |
|||
indexContent = indexContent.replace( |
|||
/<script src="registerSW\.js"><\/script>/, |
|||
'<script>/* Service worker disabled in Electron */</script>' |
|||
); |
|||
} |
|||
|
|||
// Create a temp file with modified content
|
|||
const tempDir = app.getPath('temp'); |
|||
const tempIndexPath = path.join(tempDir, 'timesafari-index.html'); |
|||
fs.writeFileSync(tempIndexPath, indexContent); |
|||
|
|||
// Load the modified index.html
|
|||
mainWindow.loadFile(tempIndexPath).catch(err => { |
|||
logError('Failed to load via loadFile:', err); |
|||
|
|||
// Fallback to direct URL loading
|
|||
mainWindow.loadURL(`file://${tempIndexPath}`).catch(err2 => { |
|||
logError('Both loading methods failed:', err2); |
|||
mainWindow.loadURL('data:text/html,<h1>Error: Failed to load TimeSafari</h1><p>Please contact support.</p>'); |
|||
}); |
|||
}); |
|||
} else { |
|||
logError(`Index file not found at: ${indexPath}`); |
|||
mainWindow.loadURL('data:text/html,<h1>Error: Cannot find application</h1><p>index.html not found</p>'); |
|||
} |
|||
} catch (err) { |
|||
logError('Failed to load app:', err); |
|||
} |
|||
|
|||
// Open DevTools in development
|
|||
if (isDev) { |
|||
mainWindow.webContents.openDevTools(); |
|||
} |
|||
|
|||
mainWindow.on('closed', () => { |
|||
mainWindow = null; |
|||
}); |
|||
} |
|||
|
|||
// Handle app ready
|
|||
app.whenReady().then(createWindow); |
|||
|
|||
// Handle all windows closed
|
|||
app.on("window-all-closed", () => { |
|||
if (process.platform !== "darwin") { |
|||
app.quit(); |
|||
} |
|||
// App lifecycle events
|
|||
app.whenReady().then(() => { |
|||
logDebug(`Starting TimeSafari v${app.getVersion()}`); |
|||
|
|||
// Skip the service worker registration for file:// protocol
|
|||
process.env.ELECTRON_DISABLE_SECURITY_WARNINGS = 'true'; |
|||
|
|||
createWindow(); |
|||
|
|||
app.on('activate', () => { |
|||
if (BrowserWindow.getAllWindows().length === 0) createWindow(); |
|||
}); |
|||
}); |
|||
|
|||
app.on("activate", () => { |
|||
if (BrowserWindow.getAllWindows().length === 0) { |
|||
createWindow(); |
|||
} |
|||
app.on('window-all-closed', () => { |
|||
if (process.platform !== 'darwin') app.quit(); |
|||
}); |
|||
|
|||
// Handle any errors
|
|||
process.on("uncaughtException", (error) => { |
|||
logger.error("Uncaught Exception:", error); |
|||
// Handle uncaught exceptions
|
|||
process.on('uncaughtException', (error) => { |
|||
logError('Uncaught Exception:', error); |
|||
}); |
|||
|
@ -1,78 +1,92 @@ |
|||
const { contextBridge, ipcRenderer } = require("electron"); |
|||
|
|||
const logger = { |
|||
log: (message, ...args) => { |
|||
if (process.env.NODE_ENV !== "production") { |
|||
/* eslint-disable no-console */ |
|||
console.log(message, ...args); |
|||
/* eslint-enable no-console */ |
|||
} |
|||
}, |
|||
warn: (message, ...args) => { |
|||
if (process.env.NODE_ENV !== "production") { |
|||
/* eslint-disable no-console */ |
|||
console.warn(message, ...args); |
|||
/* eslint-enable no-console */ |
|||
} |
|||
}, |
|||
error: (message, ...args) => { |
|||
/* eslint-disable no-console */ |
|||
console.error(message, ...args); // Errors should always be logged
|
|||
/* eslint-enable no-console */ |
|||
}, |
|||
}; |
|||
|
|||
// Use a more direct path resolution approach
|
|||
const getPath = (pathType) => { |
|||
switch (pathType) { |
|||
case "userData": |
|||
return ( |
|||
process.env.APPDATA || |
|||
(process.platform === "darwin" |
|||
? `${process.env.HOME}/Library/Application Support` |
|||
: `${process.env.HOME}/.local/share`) |
|||
); |
|||
case "home": |
|||
return process.env.HOME; |
|||
case "appPath": |
|||
return process.resourcesPath; |
|||
default: |
|||
return ""; |
|||
// Safety wrapper for logging
|
|||
function safeLog(message) { |
|||
try { |
|||
console.log('[Preload]', message); |
|||
} catch (e) { |
|||
// Silent fail for logging
|
|||
} |
|||
}; |
|||
} |
|||
|
|||
logger.log("Preload script starting..."); |
|||
// Initialize
|
|||
safeLog('Preload script starting...'); |
|||
|
|||
try { |
|||
contextBridge.exposeInMainWorld("electronAPI", { |
|||
// Path utilities
|
|||
getPath, |
|||
|
|||
// IPC functions
|
|||
// Mock service worker registration to prevent errors
|
|||
if (window.navigator) { |
|||
// Override the service worker registration to return a fake promise that resolves with nothing
|
|||
window.navigator.serviceWorker = { |
|||
register: () => Promise.resolve({}), |
|||
getRegistration: () => Promise.resolve(null), |
|||
ready: Promise.resolve({}) |
|||
}; |
|||
} |
|||
|
|||
// Safely expose specific APIs to the renderer process
|
|||
contextBridge.exposeInMainWorld('electronAPI', { |
|||
// Basic flags/info
|
|||
isElectron: true, |
|||
|
|||
// Disable service worker in Electron
|
|||
disableServiceWorker: true, |
|||
|
|||
// Logging
|
|||
log: (message) => { |
|||
try { |
|||
console.log('[Renderer]', message); |
|||
} catch (e) { |
|||
// Silence any errors from logging
|
|||
} |
|||
}, |
|||
|
|||
// Report errors to main process
|
|||
reportError: (error) => { |
|||
try { |
|||
ipcRenderer.send('app-error', error.toString()); |
|||
} catch (e) { |
|||
console.error('Failed to report error to main process', e); |
|||
} |
|||
}, |
|||
|
|||
// Safe path handling helper (no Node modules needed)
|
|||
joinPath: (...parts) => { |
|||
return parts.join('/').replace(/\/\//g, '/'); |
|||
}, |
|||
|
|||
// Fix asset URLs
|
|||
resolveAssetUrl: (assetPath) => { |
|||
if (assetPath.startsWith('/assets/')) { |
|||
return assetPath; // Already properly formed
|
|||
} |
|||
if (assetPath.startsWith('assets/')) { |
|||
return '/' + assetPath; // Add leading slash
|
|||
} |
|||
return assetPath; |
|||
}, |
|||
|
|||
// Send messages to main process
|
|||
send: (channel, data) => { |
|||
const validChannels = ["toMain"]; |
|||
// Whitelist channels for security
|
|||
const validChannels = ['app-event', 'log-event', 'app-error']; |
|||
if (validChannels.includes(channel)) { |
|||
ipcRenderer.send(channel, data); |
|||
} |
|||
}, |
|||
|
|||
// Receive messages from main process
|
|||
receive: (channel, func) => { |
|||
const validChannels = ["fromMain"]; |
|||
const validChannels = ['app-notification', 'log-response']; |
|||
if (validChannels.includes(channel)) { |
|||
ipcRenderer.on(channel, (event, ...args) => func(...args)); |
|||
// Remove old listeners to avoid memory leaks
|
|||
ipcRenderer.removeAllListeners(channel); |
|||
// Add the new listener
|
|||
ipcRenderer.on(channel, (_, ...args) => func(...args)); |
|||
} |
|||
}, |
|||
// Environment info
|
|||
env: { |
|||
isElectron: true, |
|||
isDev: process.env.NODE_ENV === "development", |
|||
}, |
|||
// Path utilities
|
|||
getBasePath: () => { |
|||
return process.env.NODE_ENV === "development" ? "/" : "./"; |
|||
}, |
|||
} |
|||
}); |
|||
|
|||
logger.log("Preload script completed successfully"); |
|||
} catch (error) { |
|||
logger.error("Error in preload script:", error); |
|||
safeLog('Preload script completed successfully'); |
|||
} catch (err) { |
|||
safeLog('Error in preload script: ' + err.toString()); |
|||
} |
|||
|
Loading…
Reference in new issue