UNPKG

@julesl23/s5js

Version:

Enhanced TypeScript SDK for S5 decentralized storage with path-based API, media processing, and directory utilities

317 lines 10.9 kB
/** * Browser compatibility detection and strategy selection */ export class BrowserCompat { static capabilities; static browserInfo; /** * Reset cached capabilities (mainly for testing) */ static resetCache() { this.capabilities = undefined; this.browserInfo = undefined; } /** * Check browser capabilities */ static async checkCapabilities() { if (this.capabilities) { return this.capabilities; } const caps = { webAssembly: false, webAssemblyStreaming: false, sharedArrayBuffer: false, webWorkers: false, offscreenCanvas: false, webP: false, avif: false, createImageBitmap: false, webGL: false, webGL2: false, memoryLimit: 512, // Default 512MB performanceAPI: false, memoryInfo: false }; // Check WebAssembly support try { if (typeof WebAssembly === 'object' && WebAssembly !== null) { caps.webAssembly = true; caps.webAssemblyStreaming = typeof WebAssembly.instantiateStreaming === 'function'; } } catch { // WebAssembly not supported } // Check SharedArrayBuffer (may be disabled due to Spectre mitigations) try { if (typeof SharedArrayBuffer !== 'undefined') { new SharedArrayBuffer(1); caps.sharedArrayBuffer = true; } } catch { // SharedArrayBuffer not supported or disabled } // Check Web Workers caps.webWorkers = typeof Worker !== 'undefined'; // Check OffscreenCanvas caps.offscreenCanvas = typeof OffscreenCanvas !== 'undefined'; // Check createImageBitmap caps.createImageBitmap = typeof createImageBitmap === 'function'; // Check WebGL support if (typeof document !== 'undefined') { try { const canvas = document.createElement('canvas'); const gl = canvas.getContext('webgl') || canvas.getContext('experimental-webgl'); caps.webGL = !!gl; const gl2 = canvas.getContext('webgl2'); caps.webGL2 = !!gl2; } catch { // WebGL not supported } } // Check Performance API caps.performanceAPI = typeof performance !== 'undefined' && typeof performance.now === 'function'; // Check memory constraints caps.memoryLimit = this.detectMemoryLimit(); caps.memoryInfo = typeof performance !== 'undefined' && !!performance.memory; // Check image format support if (this.isBrowserEnvironment()) { caps.webP = await this.checkImageFormatSupport('image/webp'); caps.avif = await this.checkImageFormatSupport('image/avif'); } this.capabilities = caps; return caps; } /** * Check if a specific image format is supported */ static checkImageFormatSupport(mimeType) { return new Promise((resolve) => { // In Node.js environment, return false if (!this.isBrowserEnvironment()) { resolve(false); return; } const img = new Image(); img.onload = () => resolve(true); img.onerror = () => resolve(false); // 1x1 pixel test images if (mimeType === 'image/webp') { // Minimal WebP image img.src = ''; } else if (mimeType === 'image/avif') { // Minimal AVIF image img.src = ''; } else { resolve(false); } }); } /** * Detect available memory limit */ static detectMemoryLimit() { // In Node.js, use process.memoryUsage if (this.isNodeEnvironment()) { try { const usage = process.memoryUsage(); return Math.floor(usage.heapTotal / 1048576); // Convert to MB } catch { return 512; // Default } } // In browser, try to use performance.memory (Chrome only) if (typeof performance !== 'undefined' && performance.memory) { const memory = performance.memory; if (memory.jsHeapSizeLimit) { return Math.floor(memory.jsHeapSizeLimit / 1048576); // Convert to MB } } // Try to estimate based on navigator.deviceMemory (Chrome only) if (typeof navigator !== 'undefined' && navigator.deviceMemory) { return navigator.deviceMemory * 1024; // Convert GB to MB } // Default fallback return 512; // 512MB default } /** * Select optimal processing strategy based on capabilities */ static selectProcessingStrategy(caps) { // Consider memory constraints - avoid WASM with very low memory const lowMemory = caps.memoryLimit < 512; // Best: WASM in Web Worker if (caps.webAssembly && caps.webWorkers && !lowMemory) { return 'wasm-worker'; } // Good: WASM in main thread if (caps.webAssembly && !lowMemory) { return 'wasm-main'; } // OK: Canvas in Web Worker if (caps.webWorkers && caps.offscreenCanvas) { return 'canvas-worker'; } // Fallback: Canvas in main thread return 'canvas-main'; } /** * Get browser information */ static getBrowserInfo() { if (this.browserInfo) { return this.browserInfo; } const userAgent = this.getUserAgent(); this.browserInfo = this.parseBrowserInfo(userAgent); return this.browserInfo; } /** * Parse browser info from user agent string */ static parseBrowserInfo(userAgent) { const info = { name: 'Unknown', version: '0', platform: 'Unknown', isMobile: false }; // Detect mobile info.isMobile = /Mobile|Android|iPhone|iPad|iPod/i.test(userAgent); // Detect platform - iOS first since it contains "Mac OS X" in user agent if (/iPhone|iPad|iPod/i.test(userAgent)) { info.platform = 'iOS'; } else if (/Android/i.test(userAgent)) { info.platform = 'Android'; } else if (/Mac OS X/i.test(userAgent)) { info.platform = 'macOS'; } else if (/Windows/i.test(userAgent)) { info.platform = 'Windows'; } else if (/Linux/i.test(userAgent)) { info.platform = 'Linux'; } // Detect browser - order matters! if (/Edg\/(\d+\.\d+\.\d+\.\d+)/i.test(userAgent)) { info.name = 'Edge'; info.version = RegExp.$1; } else if (/Chrome\/(\d+\.\d+\.\d+\.\d+)/i.test(userAgent)) { info.name = 'Chrome'; info.version = RegExp.$1; } else if (/Firefox\/(\d+\.\d+)/i.test(userAgent)) { info.name = 'Firefox'; info.version = RegExp.$1; } else if (/Version\/(\d+\.\d+\.\d+).*Safari/i.test(userAgent)) { info.name = 'Safari'; info.version = RegExp.$1; } else if (/Safari/i.test(userAgent)) { info.name = 'Safari'; // Try to extract version from Version/ tag const versionMatch = userAgent.match(/Version\/(\d+\.\d+)/); if (versionMatch) { info.version = versionMatch[1]; } } return info; } /** * Get user agent string */ static getUserAgent() { if (typeof navigator !== 'undefined' && navigator.userAgent) { return navigator.userAgent; } return ''; } /** * Get optimization recommendations based on capabilities */ static getOptimizationRecommendations(caps) { const recommendations = []; if (!caps.webAssembly) { recommendations.push('Consider upgrading to a browser with WASM support for better performance'); } if (!caps.webWorkers) { recommendations.push('Web Workers are not available - processing will block the main thread'); } if (!caps.sharedArrayBuffer) { recommendations.push('SharedArrayBuffer is disabled - parallel processing capabilities are limited'); } if (caps.memoryLimit < 512) { recommendations.push('Low memory detected - consider closing other applications'); } if (!caps.webP) { recommendations.push('WebP format not supported - using fallback formats'); } if (!caps.avif) { recommendations.push('AVIF format not supported - using older formats'); } if (!caps.offscreenCanvas) { recommendations.push('OffscreenCanvas not available - worker-based rendering is limited'); } return recommendations; } /** * Get preferred image formats based on support */ static getPreferredImageFormats(caps) { const formats = []; // Add in order of preference if (caps.avif) { formats.push('avif'); } if (caps.webP) { formats.push('webp'); } // Always include fallbacks formats.push('jpeg'); formats.push('png'); return formats; } /** * Check if running in Node.js environment */ static isNodeEnvironment() { return typeof process !== 'undefined' && process.versions != null && process.versions.node != null; } /** * Check if running in browser environment */ static isBrowserEnvironment() { return typeof window !== 'undefined' && typeof document !== 'undefined' && !this.isNodeEnvironment(); } /** * Check if running in service worker context */ static isServiceWorkerContext() { return typeof self !== 'undefined' && 'ServiceWorkerGlobalScope' in self; } /** * Check if running in web worker context */ static isWebWorkerContext() { return typeof self !== 'undefined' && typeof globalThis.importScripts === 'function' && !this.isServiceWorkerContext(); } } //# sourceMappingURL=browser.js.map