| |
| |
| |
| |
| |
|
|
| """Updates the python binaries that we cache store at |
| http://storage.google.com/webassembly. |
| |
| We only supply binaries for windows and macOS, but we do it very different ways for those two OSes. |
| On Linux, we depend on the system version of python. |
| |
| Windows recipe: |
| 1. Download precompiled version of python from NuGet package manager, |
| either the package "python" for AMD64, or "pythonarm64" for ARM64. |
| 2. Set up pip and install pywin32 and psutil via pip for emrun to work. |
| 3. Re-zip and upload to storage.google.com |
| |
| macOS recipe: |
| 1. Clone cpython |
| 2. Use homebrew to install and configure openssl (for static linking!) |
| 3. Build cpython from source and use `make install` to create archive. |
| """ |
|
|
| import glob |
| import multiprocessing |
| import os |
| import platform |
| import shutil |
| import sys |
| import urllib.request |
| from subprocess import check_call |
|
|
| from zip import unzip_cmd, zip_cmd |
|
|
| version = '3.13.3' |
| major_minor_version = '.'.join(version.split('.')[:2]) |
| |
| |
| revision = '0' |
|
|
| PSUTIL = 'psutil==7.0.0' |
|
|
| upload_base = 'gs://webassembly/emscripten-releases-builds/deps/' |
|
|
|
|
| |
| |
| def find_python_arch(): |
| import sysconfig |
| arch = sysconfig.get_platform().lower() |
| if 'amd64' in arch: |
| return 'amd64' |
| if 'arm64' in arch: |
| return 'arm64' |
| raise f'Unknown Python sysconfig platform "{arch}" (neither AMD64 or ARM64)' |
|
|
|
|
| def make_python_patch(): |
| python_arch = find_python_arch() |
| package_name = 'pythonarm64' if python_arch == 'arm64' else 'python' |
| download_url = f'https://www.nuget.org/api/v2/package/{package_name}/{version}' |
| filename = f'python-{version}-win-{python_arch}.zip' |
| out_filename = f'python-{version}-{revision}-win-{python_arch}.zip' |
|
|
| if not os.path.exists(filename): |
| print(f'Downloading python: {download_url} to {filename}') |
| urllib.request.urlretrieve(download_url, filename) |
|
|
| os.mkdir('python-nuget') |
| check_call(unzip_cmd() + [os.path.abspath(filename)], cwd='python-nuget') |
| os.remove(filename) |
|
|
| src_dir = os.path.join('python-nuget', 'tools') |
| python_exe = os.path.join(src_dir, 'python.exe') |
| check_call([python_exe, '-m', 'ensurepip', '--upgrade']) |
| check_call([python_exe, '-m', 'pip', 'install', 'pywin32==310', '--no-warn-script-location']) |
| check_call([python_exe, '-m', 'pip', 'install', PSUTIL]) |
|
|
| check_call(zip_cmd() + [os.path.join('..', '..', out_filename), '.'], cwd=src_dir) |
| print('Created: %s' % out_filename) |
|
|
| |
| shutil.rmtree('python-nuget') |
|
|
| if '--upload' in sys.argv: |
| upload_url = upload_base + out_filename |
| print('Uploading: ' + upload_url) |
| cmd = ['gsutil', 'cp', '-n', out_filename, upload_url] |
| print(' '.join(cmd)) |
| check_call(cmd) |
|
|
|
|
| def build_python(): |
| if sys.platform.startswith('darwin'): |
| |
| |
| osname = 'macos' |
| check_call(['brew', 'install', 'openssl', 'xz', 'pkg-config']) |
| if platform.machine() == 'x86_64': |
| prefix = '/usr/local' |
| min_macos_version = '11.0' |
| elif platform.machine() == 'arm64': |
| prefix = '/opt/homebrew' |
| min_macos_version = '11.0' |
|
|
| |
| |
| osname += '-' + platform.machine() |
|
|
| for f in [os.path.join(prefix, 'lib', 'libintl.dylib'), |
| os.path.join(prefix, 'include', 'libintl.h'), |
| os.path.join(prefix, 'opt', 'xz', 'lib', 'liblzma.dylib'), |
| os.path.join(prefix, 'opt', 'openssl', 'lib', 'libssl.dylib'), |
| os.path.join(prefix, 'opt', 'openssl', 'lib', 'libcrypto.dylib')]: |
| if os.path.exists(f): |
| os.remove(f) |
| os.environ['PKG_CONFIG_PATH'] = os.path.join(prefix, 'opt', 'openssl', 'lib', 'pkgconfig') |
| else: |
| osname = 'linux' |
|
|
| src_dir = 'cpython' |
| if os.path.exists(src_dir): |
| check_call(['git', 'fetch'], cwd=src_dir) |
| else: |
| check_call(['git', 'clone', 'https://github.com/python/cpython']) |
| check_call(['git', 'checkout', 'v' + version], cwd=src_dir) |
|
|
| env = os.environ |
| if sys.platform.startswith('darwin'): |
| |
| min_macos_version_line = '-mmacosx-version-min=' + min_macos_version |
| build_flags = min_macos_version_line + ' -Werror=partial-availability' |
| |
| env = env.copy() |
| env['MACOSX_DEPLOYMENT_TARGET'] = min_macos_version |
| configure_args = ['CFLAGS=' + build_flags, 'CXXFLAGS=' + build_flags, 'LDFLAGS=' + min_macos_version_line] |
| else: |
| configure_args = [] |
| check_call(['./configure'] + configure_args, cwd=src_dir, env=env) |
| check_call(['make', '-j', str(multiprocessing.cpu_count())], cwd=src_dir, env=env) |
| check_call(['make', 'install', 'DESTDIR=install'], cwd=src_dir, env=env) |
|
|
| install_dir = os.path.join(src_dir, 'install') |
|
|
| |
| |
| pybin = os.path.join(src_dir, 'install', 'usr', 'local', 'bin', 'python3') |
| pip = os.path.join(src_dir, 'install', 'usr', 'local', 'bin', 'pip3') |
| check_call([pybin, '-m', 'ensurepip', '--upgrade']) |
| check_call([pybin, pip, 'install', 'requests==2.32.3']) |
|
|
| |
| |
| check_call([pybin, pip, 'install', PSUTIL]) |
|
|
| dirname = 'python-%s-%s' % (version, revision) |
| if os.path.isdir(dirname): |
| print('Erasing old build directory ' + dirname) |
| shutil.rmtree(dirname) |
| os.rename(os.path.join(install_dir, 'usr', 'local'), dirname) |
| tarball = 'python-%s-%s-%s.tar.gz' % (version, revision, osname) |
| shutil.rmtree(os.path.join(dirname, 'lib', 'python' + major_minor_version, 'test')) |
| shutil.rmtree(os.path.join(dirname, 'include')) |
| for lib in glob.glob(os.path.join(dirname, 'lib', 'lib*.a')): |
| os.remove(lib) |
| check_call(['tar', 'zcvf', tarball, dirname]) |
|
|
| print('Created: %s' % tarball) |
| if '--upload' in sys.argv: |
| print('Uploading: ' + upload_base + tarball) |
| check_call(['gsutil', 'cp', '-n', tarball, upload_base + tarball]) |
|
|
|
|
| def main(): |
| if sys.platform.startswith('win') or '--win32' in sys.argv: |
| make_python_patch() |
| else: |
| build_python() |
| return 0 |
|
|
|
|
| if __name__ == '__main__': |
| sys.exit(main()) |
|
|