"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || (function () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.registerFileHandlers = registerFileHandlers; const electron_1 = require("electron"); const fs = __importStar(require("fs")); const path = __importStar(require("path")); const os = __importStar(require("os")); const crypto = __importStar(require("crypto")); const child_process_1 = require("child_process"); const axios_1 = __importDefault(require("axios")); const file_utils_1 = require("./file_utils"); let endpoints_in_progress = []; function registerFileHandlers() { // Flexible organization: [root]/[prefix_len-char-prefix]/[hash].file function get_organized_hashed_path(root, hash, prefix_len = 2) { const expanded_root = (0, file_utils_1.expandPath)(root); const prefix = hash.substring(0, Math.max(1, Math.min(prefix_len, 8))); const dir = path.join(expanded_root, prefix); if (!fs.existsSync(dir)) fs.mkdirSync(dir, { recursive: true }); return path.join(dir, `${hash}.file`); } electron_1.ipcMain.handle('native:check-cache', async (event, { cache_root, hash, hash_prefix_length = 2, verify_hash = false }) => { const full_path = get_organized_hashed_path(cache_root, hash, hash_prefix_length); if (!fs.existsSync(full_path)) return false; if (verify_hash) { try { const file_buffer = fs.readFileSync(full_path); const actual_hash = crypto.createHash('sha256').update(file_buffer).digest('hex'); return actual_hash === hash; } catch (e) { return false; } } return true; }); electron_1.ipcMain.handle('native:download-to-cache', async (event, { url, cache_root, hash, api_key, account_id, hash_prefix_length = 2 }) => { const full_path = get_organized_hashed_path(cache_root, hash, hash_prefix_length); const tmp_path = `${full_path}.tmp`; if (endpoints_in_progress.includes(url)) return { success: true, status: 'in_progress' }; // 1. If final file exists, skip if (fs.existsSync(full_path)) return { success: true, path: full_path, status: 'exists' }; // 2. Handle stale .tmp files (Legacy "Trust No One" pattern) if (fs.existsSync(tmp_path)) { const stats = fs.statSync(tmp_path); const age_ms = Date.now() - stats.mtimeMs; // If the tmp file is older than 5 minutes, assume previous download crashed and delete it if (age_ms > 5 * 60 * 1000) { console.log(`Native: Deleting stale temp file (${Math.round(age_ms / 1000)}s old)`); fs.unlinkSync(tmp_path); } else { return { success: true, status: 'in_progress', detail: 'fresh_tmp_exists' }; } } console.log(`Native: Hardened Download -> ${full_path}`); try { endpoints_in_progress.push(url); const response = await (0, axios_1.default)({ method: 'get', url, responseType: 'stream', headers: { 'x-aether-api-key': api_key, 'x-account-id': account_id || '' } }); const writer = fs.createWriteStream(tmp_path); response.data.pipe(writer); await new Promise((resolve, reject) => { writer.on('finish', () => resolve()); writer.on('error', reject); }); // 3. Verify Integrity before renaming (The "Trust No One" Check) const file_buffer = fs.readFileSync(tmp_path); const actual_hash = crypto.createHash('sha256').update(file_buffer).digest('hex'); if (actual_hash !== hash) { console.error(`Native: Hash Mismatch! Expected ${hash}, got ${actual_hash}`); fs.unlinkSync(tmp_path); return { success: false, error: 'Integrity check failed: Hash mismatch' }; } fs.renameSync(tmp_path, full_path); console.log(`Native: Cache Integrity Verified. File moved to final destination.`); return { success: true, path: full_path }; } catch (error) { if (fs.existsSync(tmp_path)) fs.unlinkSync(tmp_path); return { success: false, error: error.message }; } finally { endpoints_in_progress = endpoints_in_progress.filter(e => e !== url); } }); electron_1.ipcMain.handle('native:launch-from-cache', async (event, { cache_root, hash, temp_root, filename, hash_prefix_length = 2, native_template = null }) => { try { const source = get_organized_hashed_path(cache_root, hash, hash_prefix_length); const expanded_temp = (0, file_utils_1.expandPath)(temp_root); const target = path.join(expanded_temp, filename); console.log(`Native: Launching from Cache -> ${filename}`); if (!fs.existsSync(source)) { return { success: false, error: `File not in cache: ${hash}` }; } if (!fs.existsSync(expanded_temp)) fs.mkdirSync(expanded_temp, { recursive: true }); // 1. Copy the file to temp folder with original name fs.copyFileSync(source, target); // 2a. Data-driven launcher template (no rebuild needed for config changes). // Svelte resolves a Launch Profile to a single native_template string. // Format: AppleScript string with {{path}} placeholder, OR "shell: {{path}}" if (!native_template) { return { success: false, error: 'No native template configured for this file' }; } const resolved = native_template.replace(/\{\{path\}\}/g, target); if (resolved.startsWith('shell:')) { const cmd = resolved.slice(6).trim(); console.log(`Native: Running custom shell template for ${filename}`); return new Promise((resolve_fn) => { (0, child_process_1.exec)(cmd, (err, stdout, stderr) => { if (err) resolve_fn({ success: false, error: err.message, stderr: stderr.trim() }); else resolve_fn({ success: true, stdout: stdout.trim() }); }); }); } console.log(`Native: Running custom AppleScript template for ${filename}`); const tmp_script_path = path.join(os.tmpdir(), `ae_launch_${Date.now()}.scpt`); return new Promise((resolve_fn) => { try { fs.writeFileSync(tmp_script_path, resolved.trim()); } catch (e) { resolve_fn({ success: false, error: `Failed to write AppleScript temp file: ${e.message}` }); return; } (0, child_process_1.exec)(`osascript "${tmp_script_path}"`, (err) => { try { fs.unlinkSync(tmp_script_path); } catch { } if (err) resolve_fn({ success: false, error: err.message }); else resolve_fn({ success: true }); }); }); } catch (error) { return { success: false, error: error.message }; } }); // Thin primitive: copy a cached file to the temp directory with its original filename, // then return the resolved path. The caller (Svelte side) decides what to do next — // run_osascript, run_cmd, open_local_file, etc. // // This is the preferred building block for custom launch flows. Use launch_from_cache // when Svelte has already resolved a native template; use copy_from_cache_to_temp when // you want full control over what happens after the file lands in temp. electron_1.ipcMain.handle('native:copy-from-cache-to-temp', async (event, { cache_root, hash, temp_root, filename, hash_prefix_length = 2 }) => { try { const source = get_organized_hashed_path(cache_root, hash, hash_prefix_length); if (!fs.existsSync(source)) { return { success: false, error: `File not in cache: ${hash}` }; } const expanded_temp = (0, file_utils_1.expandPath)(temp_root); const target = path.join(expanded_temp, filename); if (!fs.existsSync(expanded_temp)) fs.mkdirSync(expanded_temp, { recursive: true }); fs.copyFileSync(source, target); console.log(`Native: Copied from cache to temp -> ${target}`); return { success: true, path: target }; } catch (error) { return { success: false, error: error.message }; } }); } //# sourceMappingURL=file_handlers.js.map