sneedium/main.js

413 lines
No EOL
13 KiB
JavaScript

// Modules to control application life and create native browser window
const {app, BrowserWindow, session, ipcMain, systemPreferences, dialog} = require('electron')
const path = require('path')
const fetch = require("cross-fetch")
const { ElectronChromeExtensions } = require('electron-chrome-extensions')
const { ElectronBlocker } = require('@ghostery/adblocker-electron');
const http = require('http');
const fs = require('fs');
const { createProxy } = require('proxy');
const buildChromeContextMenu = require('electron-chrome-context-menu').default
const { installCrxExtension } = require('./crx-extractor');
var extensions
var mic
var cam
// Define extension directory path
const homePath = app.getPath('home');
const extensionsDir = path.join(homePath, 'SneedExtensions');
// Ensure extensions directory exists
if (!fs.existsSync(extensionsDir)) {
fs.mkdirSync(extensionsDir, { recursive: true });
}
// IPC handlers for extension management
ipcMain.handle('get-extensions-list', async () => {
try {
const extensionFolders = fs.readdirSync(extensionsDir, { withFileTypes: true })
.filter(dirent => dirent.isDirectory())
.map(dirent => {
const extPath = path.join(extensionsDir, dirent.name);
let manifest = {};
try {
const manifestPath = path.join(extPath, 'manifest.json');
if (fs.existsSync(manifestPath)) {
manifest = JSON.parse(fs.readFileSync(manifestPath, 'utf8'));
}
} catch (e) {
console.error(`Error reading manifest for extension ${dirent.name}:`, e);
}
return {
id: dirent.name,
name: manifest.name || dirent.name,
version: manifest.version || 'unknown',
description: manifest.description || '',
path: extPath
};
});
return extensionFolders;
} catch (error) {
console.error('Error getting extensions list:', error);
return [];
}
});
ipcMain.handle('install-extension', async (event, crxPath) => {
try {
// If no path provided, open dialog to select CRX file
if (!crxPath) {
const result = await dialog.showOpenDialog({
properties: ['openFile'],
filters: [{ name: 'Chrome Extension', extensions: ['crx'] }]
});
if (result.canceled || result.filePaths.length === 0) {
return { success: false, message: 'No file selected' };
}
crxPath = result.filePaths[0];
}
// Use the CRX extractor to install the extension
const installResult = await installCrxExtension(crxPath, extensionsDir);
// If installation was successful, reload the extension
if (installResult.success && extensions) {
try {
const loadResult = extensions.loadExtension(path.join(extensionsDir, installResult.extensionId));
console.log(`Loaded extension ${installResult.extensionId}:`, loadResult);
} catch (loadError) {
console.error(`Error loading extension ${installResult.extensionId}:`, loadError);
}
}
return installResult;
} catch (error) {
console.error('Error installing extension:', error);
return { success: false, message: error.message };
}
});
ipcMain.handle('remove-extension', async (event, extensionId) => {
try {
const extensionPath = path.join(extensionsDir, extensionId);
if (fs.existsSync(extensionPath)) {
fs.rmSync(extensionPath, { recursive: true, force: true });
return { success: true };
} else {
return { success: false, message: 'Extension not found' };
}
} catch (error) {
console.error('Error removing extension:', error);
return { success: false, message: error.message };
}
});
// Handle opening extension management page
ipcMain.on('open-extensions-page', (event) => {
createExtensionsWindow();
});
app.on('web-contents-created', (event, webContents) => {
webContents.on('context-menu', (e, params) => {
const menu = buildChromeContextMenu({
params,
webContents,
openLink: (url, disposition) => {
webContents.loadURL(url)
}
})
menu.popup()
})
})
// Create an extensions management window
function createExtensionsWindow() {
const extensionsWindow = new BrowserWindow({
width: 800,
height: 600,
title: 'Extension Manager',
webPreferences: {
preload: path.join(__dirname, 'extensions-preload.js'),
nodeIntegration: false,
contextIsolation: true
}
});
extensionsWindow.loadFile('extensions.html');
extensionsWindow.setMenuBarVisibility(false);
}
// Original IPC handlers
ipcMain.on('windowmaker', (event, arg) => {
createWindow();
})
ipcMain.on('allowCam', (event, arg) => {
mic = systemPreferences.askForMediaAccess('microphone');
cam = systemPreferences.askForMediaAccess('camera');
})
const proxy = createProxy(http.createServer());
proxy.listen(3129)
//Function to enable AD Blocking...
let blocker = undefined
async function enableGoodies(s) {
try {
if (!s) {
console.log("No session provided to enableGoodies, using default session");
s = session.defaultSession;
}
blocker = await ElectronBlocker.fromLists(fetch, [
"https://github.com/uBlockOrigin/uAssets/raw/refs/heads/master/filters/ubol-filters.txt",
'https://easylist.to/easylist/easylist.txt',
'https://secure.fanboy.co.nz/fanboy-annoyance.txt',
'https://easylist.to/easylist/easyprivacy.txt',
'https://easylist-downloads.adblockplus.org/antiadblockfilters.txt',
'https://raw.githubusercontent.com/hoshsadiq/adblock-nocoin-list/master/nocoin.txt',
'https://cdn.jsdelivr.net/gh/hagezi/dns-blocklists@latest/adblock/pro.plus.txt',
"https://github.com/yokoffing/filterlists/raw/refs/heads/main/youtube_clear_view.txt",
"https://pgl.yoyo.org/as/serverlist.php?showintro=0;hostformat=adblock",
"https://github.com/uBlockOrigin/uAssets/raw/refs/heads/master/filters/unbreak.txt",
"https://github.com/uBlockOrigin/uAssets/raw/refs/heads/master/filters/quick-fixes.txt",
"https://github.com/uBlockOrigin/uAssets/raw/refs/heads/master/filters/privacy.txt",
"https://github.com/uBlockOrigin/uAssets/raw/refs/heads/master/filters/badware.txt",
"https://github.com/uBlockOrigin/uAssets/raw/refs/heads/master/filters/filters.txt"
]);
blocker.enableBlockingInSession(s);
console.log("AD blocking enabled successfully");
} catch (error) {
console.error("Error enabling AD blocking:", error);
}
}
// 0.0.0.0 day fix
const locals = [
'0.0.0.0', '127.0.0.1', '192.168', '.local', '.example', '.staging', 'fe80::', '::1'
];
// Function to check if a URL is restricted
function isLocal(url) {
return locals.some(local => url.includes(local));
}
function createWindow () {
try {
// Initialize extensions once if not already initialized
if (!extensions) {
extensions = new ElectronChromeExtensions({
// Specify extension session
session: session.defaultSession,
// Set to true to enable background pages for extensions
createBackgroundPages: true
});
}
} catch (error) {
console.error("Error initializing extensions:", error);
}
const mainWindow = new BrowserWindow({
width: 1220,
height: 600,
minWidth: 1220,
minHeight: 600,
webPreferences: {
preload: path.join(__dirname, 'preload.js'),
webviewTag: true,
// Enable devTools for development
devTools: process.env.NODE_ENV === 'development',
nodeIntegration: false,
sandbox: true,
contextIsolation: true,
// Enable Chrome extension support
additionalArguments: ['--enable-features=ExtensionsToolbarMenu'],
// Allow access to chrome:// URLs for extensions
allowRunningInsecureContent: false,
webSecurity: true
}
})
// Load installed extensions
loadExtensions(extensions);
// Register window with extensions system
if (extensions) {
extensions.addTab(mainWindow.webContents, mainWindow);
}
mainWindow.setMinimumSize(1000, 300)
const toBlock = [
"*://*.doubleclick.*",
"*://s.innovid.com/*",
"*://partner.googleadservices.com/*",
"*://*.googlesyndication.com/*",
"*://*.google-analytics.com/*",
"*://creative.ak.fbcdn.net/*",
"*://*.adbrite.com/*",
"*://*.exponential.com/*",
"*://*.quantserve.com/*",
"*://*.scorecardresearch.com/*",
"*://*.zedo.com/*",
"*://*.a-ads.com/*",
"*://*.777partner.com/*",
"*://*.77tracking.com/*",
"*://*.abc-ads.com/*",
"*://*.aaxads.com/*",
"*://*.adizio.com/*",
"*://*.adjix.com/*",
"*://*.adjug.com/*",
"*://*.adjuggler.com/*",
"*://*.trafficjunky.net/*",
"*://*.trafficleader.com/*",
"*://*.trafficrouter.io/*",
"*://*.monerominer.rocks/*",
"*://*.2mdn.net/*",
"*.exe",
"*.vbs",
"*://*.googlesyndication.*",
"*pixels*",
"*telemetry*",
"*analytics*",
"*://ads.*.com*",
"*ae/us/audience*",
"*/api/v*/science*",
"*/api/v*/typing*"
]
const regexPatterns = [
"r[0-100]+---sn-.*\.googlevideo\.com$/g",
"r[0-100]+-sn-.*\.googlevideo\.com$/g"
]
function containsAD(url) {
var i;
for (i = 0; i < toBlock.length; i++) {
let regex = toBlock[i].replace(/\*/g, "[^ ]*");
if (url.match(regex)) {
return true;
}
}
for (i = 0; i < regexPatterns.length; i++) {
let regex = regexPatterns[i]
if (url.match(regex)) {
return true;
}
}
return false;
}
session.defaultSession.setProxy({
proxyRules: 'http=localhost:3129;https=localhost:3129',
proxyBypassRules: '<local>'
})
session.defaultSession.webRequest.onBeforeRequest((details, callback) => {
if (containsAD(details.url)) {
return callback({cancel: true})
}
const url = new URL(details.url);
const hostname = url.hostname;
const isLocalDomain = isLocal(hostname);
// Check if the request is to a local domain
if (isLocalDomain) {
// Check if the request is initiated by a remote domain
const initiator = details.initiator ? new URL(details.initiator).hostname : '';
const isInitiatorLocal = isLocal(initiator);
if (initiator && !isInitiatorLocal) {
console.log(`[W] Local domain is being accessed by external source (${initiator}), don't allow!`);
callback({ cancel: true }); // Block request to local domains from remote sources
} else {
//console.log("Local domain is not being accessed by external source, allow..."); // debug
callback({ cancel: false }); // Allow request
}
} else {
//console.log("Request is not to a local domain, allow..."); //debug
callback({ cancel: false }); // Allow non-local requests
}
})
// and load the index.html of the app.
mainWindow.loadFile('index.html')
return mainWindow;
}
// Function to load all extensions from the extensions directory
function loadExtensions(extensionsInstance) {
if (!extensionsInstance) return;
try {
// Get list of extension directories
const extensionFolders = fs.readdirSync(extensionsDir, { withFileTypes: true })
.filter(dirent => dirent.isDirectory())
.map(dirent => path.join(extensionsDir, dirent.name));
// Load each extension
for (const extensionPath of extensionFolders) {
try {
if (fs.existsSync(path.join(extensionPath, 'manifest.json'))) {
const loadResult = extensionsInstance.loadExtension(extensionPath);
console.log(`Loaded extension from ${extensionPath}`, loadResult);
} else {
console.warn(`Missing manifest.json in extension directory: ${extensionPath}`);
}
} catch (extError) {
console.error(`Failed to load extension at ${extensionPath}:`, extError);
}
}
} catch (error) {
console.error('Error loading extensions:', error);
}
}
// 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.
app.whenReady().then(() => {
let x = createWindow()
enableGoodies(session.defaultSession).then(() => {
console.log("Goodies enabled");
}).catch(error => {
console.error("Error enabling goodies:", error);
});
app.on('activate', function () {
// On macOS 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 (BrowserWindow.getAllWindows().length === 0) createWindow()
})
})
// Quit when all windows are closed, except on macOS. There, it's common
// for applications and their menu bar to stay active until the user quits
// explicitly with Cmd + Q.
app.on('window-all-closed', function () {
if (process.platform !== 'darwin') { app.quit() }
})
// In this file you can include the rest of your app's specific main process
// code. You can also put them in separate files and require them here.
// Set DNS and enable mic/cam
app.on('ready', () => {
app.configureHostResolver({
mode: 'secure',
dohServers: [
'https://dns9.quad9.net/dns-query',
'https://cloudflare-dns.com/dns-query'
]
});
});