| const packages = [ |
| 'micropip', |
| 'packaging' |
| ]; |
|
|
| |
| |
| |
| |
| const pypiPackages = []; |
|
|
| import { loadPyodide } from 'pyodide'; |
| import { setGlobalDispatcher, ProxyAgent } from 'undici'; |
| import { writeFile, readFile, copyFile, readdir, rmdir, access } from 'fs/promises'; |
|
|
| |
| |
| |
| |
| function initNetworkProxyFromEnv() { |
| |
| |
| |
| |
| const allProxy = process.env.all_proxy || process.env.ALL_PROXY; |
| const httpsProxy = process.env.https_proxy || process.env.HTTPS_PROXY; |
| const httpProxy = process.env.http_proxy || process.env.HTTP_PROXY; |
| const preferedProxy = httpsProxy || allProxy || httpProxy; |
| |
| |
| |
| |
| if (!preferedProxy || !preferedProxy.startsWith('http')) return; |
| let preferedProxyURL; |
| try { |
| preferedProxyURL = new URL(preferedProxy).toString(); |
| } catch { |
| console.warn(`Invalid network proxy URL: "${preferedProxy}"`); |
| return; |
| } |
| const dispatcher = new ProxyAgent({ uri: preferedProxyURL }); |
| setGlobalDispatcher(dispatcher); |
| console.log(`Initialized network proxy "${preferedProxy}" from env`); |
| } |
|
|
| async function downloadPackages() { |
| console.log('Setting up pyodide + micropip'); |
|
|
| let pyodide; |
| try { |
| pyodide = await loadPyodide({ |
| packageCacheDir: 'static/pyodide' |
| }); |
| } catch (err) { |
| console.error('Failed to load Pyodide:', err); |
| return; |
| } |
|
|
| const packageJson = JSON.parse(await readFile('package.json')); |
| const pyodideVersion = packageJson.dependencies.pyodide.replace('^', ''); |
|
|
| try { |
| const pyodidePackageJson = JSON.parse(await readFile('static/pyodide/package.json')); |
| const pyodidePackageVersion = pyodidePackageJson.version.replace('^', ''); |
|
|
| if (pyodideVersion !== pyodidePackageVersion) { |
| console.log('Pyodide version mismatch, removing static/pyodide directory'); |
| await rmdir('static/pyodide', { recursive: true }); |
| } |
| } catch (err) { |
| console.log('Pyodide package not found, proceeding with download.', err); |
| } |
|
|
| try { |
| console.log('Loading micropip package'); |
| await pyodide.loadPackage('micropip'); |
|
|
| const micropip = pyodide.pyimport('micropip'); |
| console.log('Downloading Pyodide packages:', packages); |
|
|
| try { |
| for (const pkg of packages) { |
| console.log(`Installing package: ${pkg}`); |
| await micropip.install(pkg); |
| } |
| } catch (err) { |
| console.error('Package installation failed:', err); |
| return; |
| } |
|
|
| console.log('Pyodide packages downloaded, freezing into lock file'); |
|
|
| try { |
| const lockFile = await micropip.freeze(); |
| await writeFile('static/pyodide/pyodide-lock.json', lockFile); |
| } catch (err) { |
| console.error('Failed to write lock file:', err); |
| } |
| } catch (err) { |
| console.error('Failed to load or install micropip:', err); |
| } |
| } |
|
|
| async function copyPyodide() { |
| console.log('Copying Pyodide files into static directory'); |
| |
| for await (const entry of await readdir('node_modules/pyodide')) { |
| await copyFile(`node_modules/pyodide/${entry}`, `static/pyodide/${entry}`); |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| async function downloadPyPIWheels() { |
| const lockPath = 'static/pyodide/pyodide-lock.json'; |
| let lockData; |
| try { |
| lockData = JSON.parse(await readFile(lockPath, 'utf-8')); |
| } catch { |
| console.warn('Could not read pyodide-lock.json, skipping PyPI wheel download'); |
| return; |
| } |
|
|
| for (const pkg of pypiPackages) { |
| console.log(`Fetching PyPI metadata for: ${pkg}`); |
| const res = await fetch(`https://pypi.org/pypi/${pkg}/json`); |
| if (!res.ok) { |
| console.error(`Failed to fetch PyPI metadata for ${pkg}: ${res.status}`); |
| continue; |
| } |
| const meta = await res.json(); |
| const version = meta.info.version; |
| const files = meta.urls || []; |
| |
| const wheel = files.find( |
| (f) => f.filename.endsWith('.whl') && f.filename.includes('py3-none-any') |
| ); |
| if (!wheel) { |
| console.warn(`No pure-Python wheel found for ${pkg}==${version}, skipping`); |
| continue; |
| } |
| const dest = `static/pyodide/${wheel.filename}`; |
| |
| try { |
| await access(dest); |
| console.log(` Already exists: ${wheel.filename}`); |
| } catch { |
| console.log(` Downloading: ${wheel.filename}`); |
| const wheelRes = await fetch(wheel.url); |
| if (!wheelRes.ok) { |
| console.error(` Failed to download ${wheel.filename}: ${wheelRes.status}`); |
| continue; |
| } |
| const buffer = Buffer.from(await wheelRes.arrayBuffer()); |
| await writeFile(dest, buffer); |
| console.log(` Saved: ${dest} (${buffer.length} bytes)`); |
| } |
|
|
| |
| const normalizedName = pkg.replace(/-/g, '_'); |
| if (!lockData.packages[normalizedName]) { |
| lockData.packages[normalizedName] = { |
| name: normalizedName, |
| version: version, |
| file_name: wheel.filename, |
| install_dir: 'site', |
| sha256: wheel.digests?.sha256 || '', |
| package_type: 'package', |
| imports: [normalizedName], |
| depends: [] |
| }; |
| console.log(` Added ${normalizedName}==${version} to pyodide-lock.json`); |
| } |
| } |
|
|
| await writeFile(lockPath, JSON.stringify(lockData, null, 2)); |
| console.log('Updated pyodide-lock.json with PyPI packages'); |
| } |
|
|
| initNetworkProxyFromEnv(); |
| await downloadPackages(); |
| await copyPyodide(); |
| await downloadPyPIWheels(); |
|
|