#!/bin/bash source /venv/main/bin/activate COMFYUI_DIR=${WORKSPACE}/ComfyUI # Packages are installed after nodes so we can fix them... APT_PACKAGES=( #"package-1" #"package-2" ) PIP_PACKAGES=( #"package-1" #"package-2" ) NODES=( "https://github.com/Fannovel16/comfyui_controlnet_aux" "https://github.com/cubiq/ComfyUI_essentials" "https://github.com/lldacing/ComfyUI_PuLID_Flux_ll" "https://github.com/Derfuu/Derfuu_ComfyUI_ModdedNodes" "https://github.com/Shakker-Labs/ComfyUI-IPAdapter-Flux" ) WORKFLOWS=( "https://gist.githubusercontent.com/robballantyne/f8cb692bdcd89c96c0bd1ec0c969d905/raw/2d969f732d7873f0e1ee23b2625b50f201c722a5/flux_dev_example.json" ) STYLE_MODELS=( "https://huggingface.co/black-forest-labs/FLUX.1-Redux-dev/resolve/main/flux1-redux-dev.safetensors" ) CLIP_VISION_MODELS=( #"https://huggingface.co/google/siglip-so400m-patch14-384/resolve/main/model.safetensors" ) CLIP_MODELS=( #"https://huggingface.co/comfyanonymous/flux_text_encoders/resolve/main/clip_l.safetensors" #"https://huggingface.co/comfyanonymous/flux_text_encoders/resolve/main/t5xxl_fp16.safetensors" ) UNET_MODELS=( ) VAE_MODELS=( ) LORA_MODELS=( "https://huggingface.co/YaroslavLinder/ISKA/resolve/main/iska.safetensors" "https://huggingface.co/YaroslavLinder/Lucy/resolve/main/lucy-model.safetensors" "https://huggingface.co/enhanceaiteam/Flux-Uncensored-V2/resolve/main/lora.safetensors" ) ESRGAN_MODELS=( "https://huggingface.co/ai-forever/Real-ESRGAN/resolve/main/RealESRGAN_x2.pth" "https://huggingface.co/ai-forever/Real-ESRGAN/resolve/main/RealESRGAN_x4.pth" "https://huggingface.co/FacehugmanIII/4x_foolhardy_Remacri/resolve/main/4x_foolhardy_Remacri.pth" "https://huggingface.co/Akumetsu971/SD_Anime_Futuristic_Armor/resolve/main/4x_NMKD-Siax_200k.pth" ) CONTROLNET_MODELS=( "https://huggingface.co/Shakker-Labs/FLUX.1-dev-ControlNet-Union-Pro/resolve/main/diffusion_pytorch_model.safetensors" ) PULID_MODELS=( "https://huggingface.co/guozinan/PuLID/resolve/main/pulid_flux_v0.9.1.safetensors" ) FLUX_IPADAPTER_MODELS=( #"https://huggingface.co/InstantX/FLUX.1-dev-IP-Adapter/resolve/main/ip-adapter.bin" ) ### DO NOT EDIT BELOW HERE UNLESS YOU KNOW WHAT YOU ARE DOING ### function provisioning_start() { provisioning_print_header provisioning_get_apt_packages provisioning_get_nodes provisioning_get_pip_packages workflows_dir="${COMFYUI_DIR}/user/default/workflows" mkdir -p "${workflows_dir}" provisioning_get_files \ "${workflows_dir}" \ "${WORKFLOWS[@]}" # Get licensed models if HF_TOKEN set & valid if provisioning_has_valid_hf_token; then UNET_MODELS+=("https://huggingface.co/Comfy-Org/flux1-dev/resolve/main/flux1-dev-fp8.safetensors") VAE_MODELS+=("https://huggingface.co/black-forest-labs/FLUX.1-dev/resolve/main/ae.safetensors") else UNET_MODELS+=("https://huggingface.co/black-forest-labs/FLUX.1-schnell/resolve/main/flux1-schnell.safetensors") VAE_MODELS+=("https://huggingface.co/black-forest-labs/FLUX.1-schnell/resolve/main/ae.safetensors") sed -i 's/flux1-dev\.safetensors/flux1-schnell.safetensors/g' "${workflows_dir}/flux_dev_example.json" fi provisioning_get_files \ "${COMFYUI_DIR}/models/checkpoints" \ "${UNET_MODELS[@]}" provisioning_get_files \ "${COMFYUI_DIR}/models/vae" \ "${VAE_MODELS[@]}" provisioning_get_files \ "${COMFYUI_DIR}/models/clip" \ "${CLIP_MODELS[@]}" provisioning_get_files \ "${COMFYUI_DIR}/models/pulid" \ "${PULID_MODELS[@]}" provisioning_get_files \ "${COMFYUI_DIR}/models/clip_vision" \ "${CLIP_VISION_MODELS[@]}" provisioning_get_files \ "${COMFYUI_DIR}/models/style_models" \ "${STYLE_MODELS[@]}" provisioning_get_files \ "${COMFYUI_DIR}/models/controlnet" \ "${CONTROLNET_MODELS[@]}" provisioning_get_files \ "${COMFYUI_DIR}/models/upscale_models" \ "${ESRGAN_MODELS[@]}" provisioning_get_files \ "${COMFYUI_DIR}/models/loras" \ "${LORA_MODELS[@]}" provisioning_get_files \ "${COMFYUI_DIR}/models/ipadapter-flux" \ "${FLUX_IPADAPTER_MODELS[@]}" provisioning_print_end } function provisioning_get_apt_packages() { if [[ -n $APT_PACKAGES ]]; then sudo $APT_INSTALL ${APT_PACKAGES[@]} fi } function provisioning_get_pip_packages() { if [[ -n $PIP_PACKAGES ]]; then pip install --no-cache-dir ${PIP_PACKAGES[@]} fi } function provisioning_get_nodes() { for repo in "${NODES[@]}"; do dir="${repo##*/}" path="${COMFYUI_DIR}/custom_nodes/${dir}" requirements="${path}/requirements.txt" if [[ -d $path ]]; then if [[ ${AUTO_UPDATE,,} != "false" ]]; then printf "Updating node: %s...\n" "${repo}" ( cd "$path" && git pull ) if [[ -e $requirements ]]; then pip install --no-cache-dir -r "$requirements" fi fi else printf "Downloading node: %s...\n" "${repo}" git clone "${repo}" "${path}" --recursive if [[ -e $requirements ]]; then pip install --no-cache-dir -r "${requirements}" fi fi done } function provisioning_get_files() { if [[ -z $2 ]]; then return 1; fi dir="$1" mkdir -p "$dir" shift arr=("$@") printf "Downloading %s model(s) to %s...\n" "${#arr[@]}" "$dir" for url in "${arr[@]}"; do printf "Downloading: %s\n" "${url}" provisioning_download "${url}" "${dir}" printf "\n" done } function provisioning_print_header() { printf "\n##############################################\n# #\n# Provisioning container #\n# #\n# This will take some time #\n# #\n# Your container will be ready on completion #\n# #\n##############################################\n\n" } function provisioning_print_end() { printf "\nProvisioning complete: Application will start now\n\n" } function provisioning_has_valid_hf_token() { [[ -n "$HF_TOKEN" ]] || return 1 url="https://huggingface.co/api/whoami-v2" response=$(curl -o /dev/null -s -w "%{http_code}" -X GET "$url" \ -H "Authorization: Bearer $HF_TOKEN" \ -H "Content-Type: application/json") # Check if the token is valid if [ "$response" -eq 200 ]; then return 0 else return 1 fi } function provisioning_has_valid_civitai_token() { [[ -n "$CIVITAI_TOKEN" ]] || return 1 url="https://civitai.com/api/v1/models?hidden=1&limit=1" response=$(curl -o /dev/null -s -w "%{http_code}" -X GET "$url" \ -H "Authorization: Bearer $CIVITAI_TOKEN" \ -H "Content-Type: application/json") # Check if the token is valid if [ "$response" -eq 200 ]; then return 0 else return 1 fi } # Download from $1 URL to $2 file path function provisioning_download() { if [[ -n $HF_TOKEN && $1 =~ ^https://([a-zA-Z0-9_-]+\.)?huggingface\.co(/|$|\?) ]]; then auth_token="$HF_TOKEN" elif [[ -n $CIVITAI_TOKEN && $1 =~ ^https://([a-zA-Z0-9_-]+\.)?civitai\.com(/|$|\?) ]]; then auth_token="$CIVITAI_TOKEN" fi if [[ -n $auth_token ]];then wget --header="Authorization: Bearer $auth_token" -qnc --content-disposition --show-progress -e dotbytes="${3:-4M}" -P "$2" "$1" else wget -qnc --content-disposition --show-progress -e dotbytes="${3:-4M}" -P "$2" "$1" fi } # Allow user to disable provisioning if they started with a script they didn't want if [[ ! -f /.noprovisioning ]]; then provisioning_start fi