Spaces:
Paused
Paused
| const packages = [ | |
| 'micropip', | |
| 'packaging', | |
| 'requests', | |
| 'beautifulsoup4', | |
| 'numpy', | |
| 'pandas', | |
| 'matplotlib', | |
| 'scikit-learn', | |
| 'scipy', | |
| 'regex', | |
| 'sympy', | |
| 'tiktoken', | |
| 'seaborn', | |
| 'pytz' | |
| ]; | |
| import { loadPyodide } from 'pyodide'; | |
| import { setGlobalDispatcher, ProxyAgent } from 'undici'; | |
| import { writeFile, readFile, copyFile, readdir, rmdir } from 'fs/promises'; | |
| /** | |
| * Loading network proxy configurations from the environment variables. | |
| * And the proxy config with lowercase name has the highest priority to use. | |
| */ | |
| function initNetworkProxyFromEnv() { | |
| // we assume all subsequent requests in this script are HTTPS: | |
| // https://cdn.jsdelivr.net | |
| // https://pypi.org | |
| // https://files.pythonhosted.org | |
| 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; | |
| /** | |
| * use only http(s) proxy because socks5 proxy is not supported currently: | |
| * @see https://github.com/nodejs/undici/issues/2224 | |
| */ | |
| 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 (e) { | |
| console.log('Pyodide package not found, proceeding with download.'); | |
| } | |
| 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'); | |
| // Copy all files from node_modules/pyodide to static/pyodide | |
| for await (const entry of await readdir('node_modules/pyodide')) { | |
| await copyFile(`node_modules/pyodide/${entry}`, `static/pyodide/${entry}`); | |
| } | |
| } | |
| initNetworkProxyFromEnv(); | |
| await downloadPackages(); | |
| await copyPyodide(); | |