| #!/bin/bash |
|
|
| source /venv/main/bin/activate |
| COMFYUI_DIR=${WORKSPACE}/ComfyUI |
|
|
| |
|
|
| APT_PACKAGES=( |
| |
| |
| ) |
|
|
| PIP_PACKAGES=( |
| "--upgrade --force-reinstall https://github.com/JamePeng/llama-cpp-python/releases/download/v0.3.23-cu130-Basic-linux-20260129/llama_cpp_python-0.3.23+cu130.basic-cp312-cp312-linux_x86_64.whl" |
| ) |
|
|
| NODES=( |
| "https://github.com/ltdrdata/ComfyUI-Manager" |
| "https://github.com/cubiq/ComfyUI_essentials" |
| "https://github.com/ltdrdata/ComfyUI-Impact-Pack" |
| "https://github.com/MoonGoblinDev/Civicomfy" |
| "https://github.com/huchukato/comfy-tagcomplete" |
| "https://github.com/Koishi-Star/Euler-Smea-Dyn-Sampler" |
| "https://github.com/huchukato/ComfyUI-QwenVL-Mod" |
| "https://github.com/ltdrdata/was-node-suite-comfyui" |
| "https://github.com/Kosinkadink/ComfyUI-VideoHelperSuite" |
| "https://github.com/rgthree/rgthree-comfy" |
| "https://github.com/yolain/ComfyUI-Easy-Use" |
| "https://github.com/kijai/ComfyUI-KJNodes" |
| "https://github.com/Fannovel16/ComfyUI-Frame-Interpolation" |
| "https://github.com/Smirnov75/ComfyUI-mxToolkit" |
| "https://github.com/princepainter/ComfyUI-PainterI2V" |
| "https://github.com/princepainter/ComfyUI-PainterI2Vadvanced" |
| "https://github.com/princepainter/ComfyUI-PainterLongVideo" |
| "https://github.com/ashtar1984/comfyui-find-perfect-resolution" |
| "https://github.com/ComfyAssets/ComfyUI_Selectors" |
| "https://github.com/city96/ComfyUI-GGUF" |
| "https://github.com/kijai/ComfyUI-MMAudio" |
| "https://github.com/GACLove/ComfyUI-VFI" |
| "https://github.com/pythongosssss/ComfyUI-Custom-Scripts" |
| "https://github.com/yuvraj108c/ComfyUI-Upscaler-Tensorrt" |
| "https://github.com/huchukato/ComfyUI-RIFE-TensorRT-Auto" |
| "https://github.com/stduhpf/ComfyUI-WanMoeKSampler" |
| "https://github.com/melMass/comfy_mtb" |
| ) |
|
|
| WORKFLOWS=( |
| "https://huggingface.co/huchukato/favs/resolve/main/workflows/SDXL-LoRaStack-Upscale.json" |
| "https://huggingface.co/huchukato/favs/resolve/main/workflows/Wan2.2-I2V-SVI-AutoPrompt-1-2.json" |
| "https://huggingface.co/huchukato/favs/resolve/main/workflows/Wan2.2-I2V-SVI-AutoPrompt-GGUF.json" |
| "https://huggingface.co/huchukato/favs/resolve/main/workflows/WAN2.2-I2V-AutoPrompt-1-4.json" |
| "https://huggingface.co/huchukato/favs/resolve/main/workflows/WAN2.2-I2V-AutoPrompt-GGUF-1-4.json" |
| "https://huggingface.co/huchukato/favs/resolve/main/workflows/WAN2.2-T2V-AutoPrompt-1-4.json" |
| "https://huggingface.co/huchukato/favs/resolve/main/workflows/WAN2.2-T2V-AutoPrompt-GGUF-1-3.json" |
| "https://huggingface.co/huchukato/favs/resolve/main/workflows/WAN2.2-I2V-Full-AutoPrompt-MMAudio-v1-7.json" |
| "https://huggingface.co/huchukato/favs/resolve/main/workflows/WAN2.2-I2V-Full-AutoPrompt-MMAudio-GGUF-v1-4.json" |
| ) |
|
|
|
|
| CHECKPOINT_MODELS=( |
| ) |
|
|
| UNET_MODELS=( |
| |
| ) |
|
|
| LORA_MODELS=( |
| |
| ) |
|
|
| VAE_MODELS=( |
| "https://huggingface.co/Comfy-Org/Wan_2.2_ComfyUI_Repackaged/resolve/main/split_files/vae/wan_2.1_vae.safetensors" |
| "https://huggingface.co/Kijai/WanVideo_comfy/resolve/main/Wan2_1_VAE_fp32.safetensors" |
| "https://huggingface.co/huchukato/favs/resolve/main/VAE/sdxl.vae.safetensors" |
| ) |
|
|
| ESRGAN_MODELS=( |
| "https://huggingface.co/huchukato/favs/resolve/main/ESRGAN/2xLexicaRRDBNet.pth" |
| "https://huggingface.co/huchukato/favs/resolve/main/ESRGAN/2xLexicaRRDBNet_Sharp.pth" |
| ) |
|
|
| TEXT_ENCODERS=( |
| "https://huggingface.co/NSFW-API/NSFW-Wan-UMT5-XXL/resolve/main/nsfw_wan_umt5-xxl_fp8_scaled.safetensors" |
| ) |
|
|
| CONTROLNET_MODELS=( |
| ) |
|
|
| |
|
|
| function provisioning_start() { |
| provisioning_print_header |
| echo "๐ Starting provisioning process..." |
| |
| echo "๐ฆ Installing APT packages..." |
| provisioning_get_apt_packages |
| |
| echo "๐ง Installing custom nodes..." |
| provisioning_get_nodes |
| |
| echo "๐ฅ Installing PIP packages..." |
| provisioning_get_pip_packages |
| |
| echo "๐ Downloading workflows..." |
| provisioning_get_files \ |
| "${COMFYUI_DIR}/user/default/workflows" \ |
| "${WORKFLOWS[@]}" |
| |
| echo "๐ฏ Downloading checkpoint models..." |
| provisioning_get_files \ |
| "${COMFYUI_DIR}/models/checkpoints" \ |
| "${CHECKPOINT_MODELS[@]}" |
| |
| echo "๐ง Downloading U-NET models..." |
| provisioning_get_files \ |
| "${COMFYUI_DIR}/models/unet" \ |
| "${UNET_MODELS[@]}" |
| |
| echo "๐จ Downloading LoRA models..." |
| provisioning_get_files \ |
| "${COMFYUI_DIR}/models/lora" \ |
| "${LORA_MODELS[@]}" |
| |
| echo "๐ฎ Downloading ControlNet models..." |
| provisioning_get_files \ |
| "${COMFYUI_DIR}/models/controlnet" \ |
| "${CONTROLNET_MODELS[@]}" |
| |
| echo "๐ฎ Downloading VAE models..." |
| provisioning_get_files \ |
| "${COMFYUI_DIR}/models/vae" \ |
| "${VAE_MODELS[@]}" |
| |
| echo "โก Downloading upscale models..." |
| provisioning_get_files \ |
| "${COMFYUI_DIR}/models/upscale_models" \ |
| "${ESRGAN_MODELS[@]}" |
| |
| echo "๐ Downloading text encoders..." |
| provisioning_get_files \ |
| "${COMFYUI_DIR}/models/text_encoders" \ |
| "${TEXT_ENCODERS[@]}" |
| |
| 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 |
| echo "Installing PIP packages..." |
| for package in "${PIP_PACKAGES[@]}"; do |
| echo "Installing: $package" |
| pip install --root-user-action=ignore --no-cache-dir $package |
| echo "โ Completed: $package" |
| done |
| echo "All PIP packages installed successfully!" |
| fi |
| } |
|
|
| function provisioning_get_nodes() { |
| echo "Processing ${#NODES[@]} nodes..." |
| local count=0 |
| for repo in "${NODES[@]}"; do |
| ((count++)) |
| dir="${repo##*/}" |
| path="${COMFYUI_DIR}/custom_nodes/${dir}" |
| requirements="${path}/requirements.txt" |
| |
| echo "[$count/${#NODES[@]}] Processing node: $dir" |
| |
| if [[ -d $path ]]; then |
| if [[ ${AUTO_UPDATE,,} != "false" ]]; then |
| echo " โ Updating existing node..." |
| ( cd "$path" && git pull ) |
| if [[ -e $requirements ]]; then |
| echo " โ Installing requirements..." |
| pip install --root-user-action=ignore --no-cache-dir -r "$requirements" |
| fi |
| else |
| echo " โ Node exists, skipping (AUTO_UPDATE=false)" |
| fi |
| else |
| echo " โ Downloading new node..." |
| git clone "${repo}" "${path}" --recursive |
| if [[ -e $requirements ]]; then |
| echo " โ Installing requirements..." |
| pip install --root-user-action=ignore --no-cache-dir -r "${requirements}" |
| fi |
| fi |
| echo " โ Node $dir completed" |
| done |
| echo "All nodes processed successfully!" |
| } |
|
|
| function provisioning_get_files() { |
| if [[ -z $2 ]]; then return 1; fi |
| |
| dir="$1" |
| mkdir -p "$dir" |
| shift |
| arr=("$@") |
| echo "Downloading ${#arr[@]} file(s) to $dir..." |
| local count=0 |
| for url in "${arr[@]}"; do |
| ((count++)) |
| echo "[$count/${#arr[@]}] Downloading: $(basename "$url")" |
| provisioning_download "${url}" "${dir}" |
| echo " โ Download completed" |
| done |
| echo "All files downloaded successfully!" |
| } |
|
|
| 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") |
|
|
| |
| 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") |
|
|
| |
| if [ "$response" -eq 200 ]; then |
| return 0 |
| else |
| return 1 |
| fi |
| } |
|
|
| |
| 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 |
| } |
|
|
| |
| if [[ ! -f /.noprovisioning ]]; then |
| provisioning_start |
| fi |