Skip to content

Commit

Permalink
speed up remote bootstrap for @pypi/@conda
Browse files Browse the repository at this point in the history
  • Loading branch information
savingoyal committed Oct 5, 2024
1 parent 7a22db0 commit 40f4b0f
Show file tree
Hide file tree
Showing 3 changed files with 129 additions and 65 deletions.
191 changes: 127 additions & 64 deletions metaflow/plugins/pypi/bootstrap.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,15 @@
import bz2
import concurrent.futures
import io
import json
import os
import shutil
import subprocess
import sys
import tarfile
import time

import requests

from metaflow.metaflow_config import DATASTORE_LOCAL_DIR
from metaflow.plugins import DATASTORES
Expand All @@ -15,7 +19,14 @@

# Bootstraps a valid conda virtual environment composed of conda and pypi packages


def print_timer(operation, start_time):
duration = time.time() - start_time
print(f"Time taken for {operation}: {duration:.2f} seconds")


if __name__ == "__main__":
total_start_time = time.time()
if len(sys.argv) != 5:
print("Usage: bootstrap.py <flow_name> <id> <datastore_type> <architecture>")
sys.exit(1)
Expand Down Expand Up @@ -47,6 +58,8 @@

prefix = os.path.join(os.getcwd(), architecture, id_)
pkgs_dir = os.path.join(os.getcwd(), ".pkgs")
conda_pkgs_dir = os.path.join(pkgs_dir, "conda")
pypi_pkgs_dir = os.path.join(pkgs_dir, "pypi")
manifest_dir = os.path.join(os.getcwd(), DATASTORE_LOCAL_DIR, flow_name)

datastores = [d for d in DATASTORES if d.TYPE == datastore_type]
Expand All @@ -64,77 +77,127 @@
os.path.join(os.getcwd(), MAGIC_FILE),
os.path.join(manifest_dir, MAGIC_FILE),
)

with open(os.path.join(manifest_dir, MAGIC_FILE)) as f:
env = json.load(f)[id_][architecture]

# Download Conda packages.
conda_pkgs_dir = os.path.join(pkgs_dir, "conda")
with storage.load_bytes([package["path"] for package in env["conda"]]) as results:
for key, tmpfile, _ in results:
# Ensure that conda packages go into architecture specific folders.
# The path looks like REPO/CHANNEL/CONDA_SUBDIR/PACKAGE. We trick
# Micromamba into believing that all packages are coming from a local
# channel - the only hurdle is ensuring that packages are organised
# properly.

# TODO: consider RAM disk
dest = os.path.join(conda_pkgs_dir, "/".join(key.split("/")[-2:]))
os.makedirs(os.path.dirname(dest), exist_ok=True)
shutil.move(tmpfile, dest)

# Create Conda environment.
cmds = [
def run_cmd(cmd):
cmd_start_time = time.time()
result = subprocess.run(
cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE, text=True
)
print_timer(f"Command: {cmd}", cmd_start_time)
if result.returncode != 0:
print(f"Bootstrap failed while executing: {cmd}")
print("Stdout:", result.stdout)
print("Stderr:", result.stderr)
sys.exit(1)

def install_micromamba(architecture):
# TODO: check if mamba or conda are already available on the image
# TODO: micromamba installation can be pawned off to micromamba.py
f"""set -e;
if ! command -v micromamba >/dev/null 2>&1; then
mkdir -p micromamba;
python -c "import requests, bz2, sys; data = requests.get('https://micro.mamba.pm/api/micromamba/{architecture}/1.5.7').content; sys.stdout.buffer.write(bz2.decompress(data))" | tar -xv -C $(pwd)/micromamba bin/micromamba --strip-components 1;
export PATH=$PATH:$(pwd)/micromamba;
if ! command -v micromamba >/dev/null 2>&1; then
echo "Failed to install Micromamba!";
exit 1;
fi;
fi""",
# Create a conda environment through Micromamba.
f'''set -e;
tmpfile=$(mktemp);
echo "@EXPLICIT" > "$tmpfile";
ls -d {conda_pkgs_dir}/*/* >> "$tmpfile";
export PATH=$PATH:$(pwd)/micromamba;
export CONDA_PKGS_DIRS=$(pwd)/micromamba/pkgs;
micromamba create --yes --offline --no-deps --safety-checks=disabled --no-extra-safety-checks --prefix {prefix} --file "$tmpfile";
rm "$tmpfile"''',
]

# Download PyPI packages.
if "pypi" in env:
pypi_pkgs_dir = os.path.join(pkgs_dir, "pypi")
with storage.load_bytes(
[package["path"] for package in env["pypi"]]
) as results:
micromamba_timer = time.time()
micromamba_dir = os.path.join(os.getcwd(), "micromamba")
micromamba_path = os.path.join(micromamba_dir, "bin", "micromamba")

if which("micromamba") or os.path.exists(micromamba_path):
return micromamba_path

os.makedirs(micromamba_dir, exist_ok=True)
# TODO: download micromamba from datastore
url = f"https://micro.mamba.pm/api/micromamba/{architecture}/1.5.7"
response = requests.get(url, stream=True)
if response.status_code != 200:
raise Exception(
f"Failed to download micromamba: HTTP {response.status_code}"
)
tar_content = bz2.BZ2Decompressor().decompress(response.raw.read())
with tarfile.open(fileobj=io.BytesIO(tar_content), mode="r:") as tar:
tar.extract("bin/micromamba", path=micromamba_dir, set_attrs=False)

os.chmod(micromamba_path, 0o755)
if not os.path.exists(micromamba_path):
raise Exception("Failed to install Micromamba!")

os.environ["PATH"] += os.pathsep + os.path.dirname(micromamba_path)
print_timer("Downloading micromamba", micromamba_timer)
return micromamba_path

def download_conda_packages(storage, packages, dest_dir):
download_start_time = time.time()
os.makedirs(dest_dir, exist_ok=True)
with storage.load_bytes([package["path"] for package in packages]) as results:
for key, tmpfile, _ in results:
dest = os.path.join(pypi_pkgs_dir, os.path.basename(key))
# Ensure that conda packages go into architecture specific folders.
# The path looks like REPO/CHANNEL/CONDA_SUBDIR/PACKAGE. We trick
# Micromamba into believing that all packages are coming from a local
# channel - the only hurdle is ensuring that packages are organised
# properly.

# TODO: consider RAM disk
dest = os.path.join(dest_dir, "/".join(key.split("/")[-2:]))
os.makedirs(os.path.dirname(dest), exist_ok=True)
shutil.move(tmpfile, dest)
print_timer("Downloading conda packages", download_start_time)
return dest_dir

# Install PyPI packages.
cmds.extend(
[
f"""set -e;
export PATH=$PATH:$(pwd)/micromamba;
export CONDA_PKGS_DIRS=$(pwd)/micromamba/pkgs;
micromamba run --prefix {prefix} python -m pip --disable-pip-version-check install --root-user-action=ignore --no-compile {pypi_pkgs_dir}/*.whl --no-user"""
]
)
def download_pypi_packages(storage, packages, dest_dir):
download_start_time = time.time()
os.makedirs(dest_dir, exist_ok=True)
with storage.load_bytes([package["path"] for package in packages]) as results:
for key, tmpfile, _ in results:
dest = os.path.join(dest_dir, os.path.basename(key))
shutil.move(tmpfile, dest)
print_timer("Downloading pypi packages", download_start_time)
return dest_dir

def create_conda_environment(prefix, conda_pkgs_dir):
cmd = f'''set -e;
tmpfile=$(mktemp);
echo "@EXPLICIT" > "$tmpfile";
ls -d {conda_pkgs_dir}/*/* >> "$tmpfile";
export PATH=$PATH:$(pwd)/micromamba;
export CONDA_PKGS_DIRS=$(pwd)/micromamba/pkgs;
micromamba create --yes --offline --no-deps --safety-checks=disabled --no-extra-safety-checks --prefix {prefix} --file "$tmpfile";
rm "$tmpfile"'''
run_cmd(cmd)

for cmd in cmds:
result = subprocess.run(
cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE
def install_pypi_packages(prefix, pypi_pkgs_dir):
cmd = f"""set -e;
export PATH=$PATH:$(pwd)/micromamba;
export CONDA_PKGS_DIRS=$(pwd)/micromamba/pkgs;
micromamba run --prefix {prefix} python -m pip --disable-pip-version-check install --root-user-action=ignore --no-compile --no-index --no-cache-dir --no-deps --prefer-binary --find-links={pypi_pkgs_dir} {pypi_pkgs_dir}/*.whl --no-user"""
run_cmd(cmd)

with concurrent.futures.ThreadPoolExecutor(max_workers=3) as executor:
# install micromamba, download conda and pypi packages in parallel
future_install_micromamba = executor.submit(install_micromamba, architecture)
future_download_conda_packages = executor.submit(
download_conda_packages, storage, env["conda"], conda_pkgs_dir
)
if result.returncode != 0:
print(f"Bootstrap failed while executing: {cmd}")
print("Stdout:", result.stdout.decode())
print("Stderr:", result.stderr.decode())
sys.exit(1)
future_download_pypi_packages = None
if "pypi" in env:
future_download_pypi_packages = executor.submit(
download_pypi_packages, storage, env["pypi"], pypi_pkgs_dir
)
# create conda environment after micromamba is installed and conda packages are downloaded
concurrent.futures.wait(
[future_install_micromamba, future_download_conda_packages]
)
future_create_conda_environment = executor.submit(
create_conda_environment, prefix, conda_pkgs_dir
)
if "pypi" in env:
# install pypi packages after conda environment is created and pypi packages are downloaded
concurrent.futures.wait(
[future_create_conda_environment, future_download_pypi_packages]
)
future_install_pypi_packages = executor.submit(
install_pypi_packages, prefix, pypi_pkgs_dir
)
# wait for pypi packages to be installed
future_install_pypi_packages.result()
else:
# wait for conda environment to be created
future_create_conda_environment.result()

total_time = time.time() - total_start_time
print(f"{total_time:.2f}")
2 changes: 1 addition & 1 deletion metaflow/plugins/pypi/conda_environment.py
Original file line number Diff line number Diff line change
Expand Up @@ -374,7 +374,7 @@ def bootstrap_commands(self, step_name, datastore_type):
'DISABLE_TRACING=True python -m metaflow.plugins.pypi.bootstrap "%s" %s "%s" linux-64'
% (self.flow.name, id_, self.datastore_type),
"echo 'Environment bootstrapped.'",
"export PATH=$PATH:$(pwd)/micromamba",
"export PATH=$PATH:$(pwd)/micromamba/bin",
]
else:
# for @conda/@pypi(disabled=True).
Expand Down
1 change: 1 addition & 0 deletions metaflow/plugins/pypi/micromamba.py
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ def __init__(self):
os.path.expanduser(_home),
"micromamba",
)

self.bin = (
which(os.environ.get("METAFLOW_PATH_TO_MICROMAMBA") or "micromamba")
or which("./micromamba") # to support remote execution
Expand Down

0 comments on commit 40f4b0f

Please sign in to comment.