commit
070c7eb728
29
README.md
29
README.md
@ -44,6 +44,17 @@ If you run on Linux and would like to use the GUI, there is now a port of it as
|
|||||||
### Runpod
|
### Runpod
|
||||||
Follow the instructions found in this discussion: https://github.com/bmaltais/kohya_ss/discussions/379
|
Follow the instructions found in this discussion: https://github.com/bmaltais/kohya_ss/discussions/379
|
||||||
|
|
||||||
|
### MacOS
|
||||||
|
In the terminal, run
|
||||||
|
|
||||||
|
```
|
||||||
|
git clone https://github.com/bmaltais/kohya_ss.git
|
||||||
|
cd kohya_ss
|
||||||
|
bash macos_setup.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
During the accelerate config screen after running the script answer "This machine", "None", "No" for the remaining questions.
|
||||||
|
|
||||||
### Ubuntu
|
### Ubuntu
|
||||||
In the terminal, run
|
In the terminal, run
|
||||||
|
|
||||||
@ -99,7 +110,17 @@ Run the following commands to install:
|
|||||||
python .\tools\cudann_1.8_install.py
|
python .\tools\cudann_1.8_install.py
|
||||||
```
|
```
|
||||||
|
|
||||||
## Upgrading
|
## Upgrading MacOS
|
||||||
|
|
||||||
|
When a new release comes out, you can upgrade your repo with the following commands in the root directory:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
upgrade_macos.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
Once the commands have completed successfully you should be ready to use the new version.
|
||||||
|
|
||||||
|
## Upgrading Windows
|
||||||
|
|
||||||
When a new release comes out, you can upgrade your repo with the following commands in the root directory:
|
When a new release comes out, you can upgrade your repo with the following commands in the root directory:
|
||||||
|
|
||||||
@ -192,6 +213,12 @@ This will store your a backup file with your current locally installed pip packa
|
|||||||
|
|
||||||
## Change History
|
## Change History
|
||||||
|
|
||||||
|
* 2023/03/25 (v21.3.4)
|
||||||
|
- Added untested support for MacOS base on this gist: https://gist.github.com/jstayco/9f5733f05b9dc29de95c4056a023d645
|
||||||
|
|
||||||
|
Let me know how this work. From the look of it it appear to be well tought out. I modified a few things to make it fit better with the rest of the code in the repo.
|
||||||
|
- Fix for issue https://github.com/bmaltais/kohya_ss/issues/433 by implementing default of 0.
|
||||||
|
- Removed non applicable save_model_as choices for LoRA and TI.
|
||||||
* 2023/03/24 (v21.3.3)
|
* 2023/03/24 (v21.3.3)
|
||||||
- Add support for custom user gui files. THey will be created at installation time or when upgrading is missing. You will see two files in the root of the folder. One named `gui-user.bat` and the other `gui-user.ps1`. Edit the file based on your prefered terminal. Simply add the parameters you want to pass the gui in there and execute it to start the gui with them. Enjoy!
|
- Add support for custom user gui files. THey will be created at installation time or when upgrading is missing. You will see two files in the root of the folder. One named `gui-user.bat` and the other `gui-user.ps1`. Edit the file based on your prefered terminal. Simply add the parameters you want to pass the gui in there and execute it to start the gui with them. Enjoy!
|
||||||
* 2023/03/23 (v21.3.2)
|
* 2023/03/23 (v21.3.2)
|
||||||
|
13
gui_macos.sh
Executable file
13
gui_macos.sh
Executable file
@ -0,0 +1,13 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Activate the virtual environment
|
||||||
|
source venv/bin/activate
|
||||||
|
|
||||||
|
# Validate the requirements and store the exit code
|
||||||
|
python tools/validate_requirements.py --requirements requirements_macos.txt
|
||||||
|
exit_code=$?
|
||||||
|
|
||||||
|
# If the exit code is 0, run the kohya_gui.py script with the command-line arguments
|
||||||
|
if [ $exit_code -eq 0 ]; then
|
||||||
|
python kohya_gui.py "$@"
|
||||||
|
fi
|
@ -1,4 +1,5 @@
|
|||||||
from tkinter import filedialog, Tk
|
from tkinter import filedialog, Tk
|
||||||
|
from easygui import msgbox
|
||||||
import os
|
import os
|
||||||
import gradio as gr
|
import gradio as gr
|
||||||
import easygui
|
import easygui
|
||||||
@ -60,28 +61,20 @@ def check_if_model_exist(output_name, output_dir, save_model_as):
|
|||||||
|
|
||||||
|
|
||||||
def update_my_data(my_data):
|
def update_my_data(my_data):
|
||||||
# Update optimizer based on use_8bit_adam flag
|
# Update the optimizer based on the use_8bit_adam flag
|
||||||
use_8bit_adam = my_data.get('use_8bit_adam', False)
|
use_8bit_adam = my_data.get('use_8bit_adam', False)
|
||||||
if use_8bit_adam:
|
my_data.setdefault('optimizer', 'AdamW8bit' if use_8bit_adam else 'AdamW')
|
||||||
my_data['optimizer'] = 'AdamW8bit'
|
|
||||||
elif 'optimizer' not in my_data:
|
|
||||||
my_data['optimizer'] = 'AdamW'
|
|
||||||
|
|
||||||
# Update model_list to custom if empty or pretrained_model_name_or_path is not a preset model
|
# Update model_list to custom if empty or pretrained_model_name_or_path is not a preset model
|
||||||
model_list = my_data.get('model_list', [])
|
model_list = my_data.get('model_list', [])
|
||||||
pretrained_model_name_or_path = my_data.get(
|
pretrained_model_name_or_path = my_data.get('pretrained_model_name_or_path', '')
|
||||||
'pretrained_model_name_or_path', ''
|
if not model_list or pretrained_model_name_or_path not in ALL_PRESET_MODELS:
|
||||||
)
|
|
||||||
if (
|
|
||||||
not model_list
|
|
||||||
or pretrained_model_name_or_path not in ALL_PRESET_MODELS
|
|
||||||
):
|
|
||||||
my_data['model_list'] = 'custom'
|
my_data['model_list'] = 'custom'
|
||||||
|
|
||||||
# Convert epoch and save_every_n_epochs values to int if they are strings
|
# Convert epoch and save_every_n_epochs values to int if they are strings
|
||||||
for key in ['epoch', 'save_every_n_epochs']:
|
for key in ['epoch', 'save_every_n_epochs']:
|
||||||
value = my_data.get(key, -1)
|
value = my_data.get(key, -1)
|
||||||
if isinstance(value, str) and value:
|
if isinstance(value, str) and value.isdigit():
|
||||||
my_data[key] = int(value)
|
my_data[key] = int(value)
|
||||||
elif not value:
|
elif not value:
|
||||||
my_data[key] = -1
|
my_data[key] = -1
|
||||||
@ -90,43 +83,23 @@ def update_my_data(my_data):
|
|||||||
if my_data.get('LoRA_type', 'Standard') == 'LoCon':
|
if my_data.get('LoRA_type', 'Standard') == 'LoCon':
|
||||||
my_data['LoRA_type'] = 'LyCORIS/LoCon'
|
my_data['LoRA_type'] = 'LyCORIS/LoCon'
|
||||||
|
|
||||||
|
# Update model save choices due to changes for LoRA and TI training
|
||||||
|
if (
|
||||||
|
(my_data.get('LoRA_type') or my_data.get('num_vectors_per_token'))
|
||||||
|
and my_data.get('save_model_as') not in ['safetensors', 'ckpt']
|
||||||
|
):
|
||||||
|
message = (
|
||||||
|
'Updating save_model_as to safetensors because the current value in the config file is no longer applicable to {}'
|
||||||
|
)
|
||||||
|
if my_data.get('LoRA_type'):
|
||||||
|
print(message.format('LoRA'))
|
||||||
|
if my_data.get('num_vectors_per_token'):
|
||||||
|
print(message.format('TI'))
|
||||||
|
my_data['save_model_as'] = 'safetensors'
|
||||||
|
|
||||||
return my_data
|
return my_data
|
||||||
|
|
||||||
|
|
||||||
# def update_my_data(my_data):
|
|
||||||
# if my_data.get('use_8bit_adam', False) == True:
|
|
||||||
# my_data['optimizer'] = 'AdamW8bit'
|
|
||||||
# # my_data['use_8bit_adam'] = False
|
|
||||||
|
|
||||||
# if (
|
|
||||||
# my_data.get('optimizer', 'missing') == 'missing'
|
|
||||||
# and my_data.get('use_8bit_adam', False) == False
|
|
||||||
# ):
|
|
||||||
# my_data['optimizer'] = 'AdamW'
|
|
||||||
|
|
||||||
# if my_data.get('model_list', 'custom') == []:
|
|
||||||
# print('Old config with empty model list. Setting to custom...')
|
|
||||||
# my_data['model_list'] = 'custom'
|
|
||||||
|
|
||||||
# # If Pretrained model name or path is not one of the preset models then set the preset_model to custom
|
|
||||||
# if not my_data.get('pretrained_model_name_or_path', '') in ALL_PRESET_MODELS:
|
|
||||||
# my_data['model_list'] = 'custom'
|
|
||||||
|
|
||||||
# # Fix old config files that contain epoch as str instead of int
|
|
||||||
# for key in ['epoch', 'save_every_n_epochs']:
|
|
||||||
# value = my_data.get(key, -1)
|
|
||||||
# if type(value) == str:
|
|
||||||
# if value != '':
|
|
||||||
# my_data[key] = int(value)
|
|
||||||
# else:
|
|
||||||
# my_data[key] = -1
|
|
||||||
|
|
||||||
# if my_data.get('LoRA_type', 'Standard') == 'LoCon':
|
|
||||||
# my_data['LoRA_type'] = 'LyCORIS/LoCon'
|
|
||||||
|
|
||||||
# return my_data
|
|
||||||
|
|
||||||
|
|
||||||
def get_dir_and_file(file_path):
|
def get_dir_and_file(file_path):
|
||||||
dir_path, file_name = os.path.split(file_path)
|
dir_path, file_name = os.path.split(file_path)
|
||||||
return (dir_path, file_name)
|
return (dir_path, file_name)
|
||||||
@ -604,7 +577,13 @@ def get_pretrained_model_name_or_path_file(
|
|||||||
set_model_list(model_list, pretrained_model_name_or_path)
|
set_model_list(model_list, pretrained_model_name_or_path)
|
||||||
|
|
||||||
|
|
||||||
def gradio_source_model():
|
def gradio_source_model(save_model_as_choices = [
|
||||||
|
'same as source model',
|
||||||
|
'ckpt',
|
||||||
|
'diffusers',
|
||||||
|
'diffusers_safetensors',
|
||||||
|
'safetensors',
|
||||||
|
]):
|
||||||
with gr.Tab('Source model'):
|
with gr.Tab('Source model'):
|
||||||
# Define the input elements
|
# Define the input elements
|
||||||
with gr.Row():
|
with gr.Row():
|
||||||
@ -646,13 +625,7 @@ def gradio_source_model():
|
|||||||
)
|
)
|
||||||
save_model_as = gr.Dropdown(
|
save_model_as = gr.Dropdown(
|
||||||
label='Save trained model as',
|
label='Save trained model as',
|
||||||
choices=[
|
choices=save_model_as_choices,
|
||||||
'same as source model',
|
|
||||||
'ckpt',
|
|
||||||
'diffusers',
|
|
||||||
'diffusers_safetensors',
|
|
||||||
'safetensors',
|
|
||||||
],
|
|
||||||
value='safetensors',
|
value='safetensors',
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -954,6 +927,7 @@ def gradio_advanced_training():
|
|||||||
max_data_loader_n_workers = gr.Textbox(
|
max_data_loader_n_workers = gr.Textbox(
|
||||||
label='Max num workers for DataLoader',
|
label='Max num workers for DataLoader',
|
||||||
placeholder='(Optional) Override number of epoch. Default: 8',
|
placeholder='(Optional) Override number of epoch. Default: 8',
|
||||||
|
value="0",
|
||||||
)
|
)
|
||||||
return (
|
return (
|
||||||
# use_8bit_adam,
|
# use_8bit_adam,
|
||||||
|
@ -257,7 +257,8 @@ def open_configuration(
|
|||||||
with open(file_path, 'r') as f:
|
with open(file_path, 'r') as f:
|
||||||
my_data = json.load(f)
|
my_data = json.load(f)
|
||||||
print('Loading config...')
|
print('Loading config...')
|
||||||
# Update values to fix deprecated use_8bit_adam checkbox and set appropriate optimizer if it is set to True
|
|
||||||
|
# Update values to fix deprecated use_8bit_adam checkbox, set appropriate optimizer if it is set to True, etc.
|
||||||
my_data = update_my_data(my_data)
|
my_data = update_my_data(my_data)
|
||||||
else:
|
else:
|
||||||
file_path = original_file_path # In case a file_path was provided and the user decide to cancel the open action
|
file_path = original_file_path # In case a file_path was provided and the user decide to cancel the open action
|
||||||
@ -648,7 +649,10 @@ def lora_tab(
|
|||||||
v_parameterization,
|
v_parameterization,
|
||||||
save_model_as,
|
save_model_as,
|
||||||
model_list,
|
model_list,
|
||||||
) = gradio_source_model()
|
) = gradio_source_model(save_model_as_choices = [
|
||||||
|
'ckpt',
|
||||||
|
'safetensors',
|
||||||
|
])
|
||||||
|
|
||||||
with gr.Tab('Folders'):
|
with gr.Tab('Folders'):
|
||||||
with gr.Row():
|
with gr.Row():
|
||||||
|
38
macos_setup.sh
Executable file
38
macos_setup.sh
Executable file
@ -0,0 +1,38 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# The initial setup script to prep the environment on macOS
|
||||||
|
# xformers has been omitted as that is for Nvidia GPUs only
|
||||||
|
|
||||||
|
if ! command -v brew >/dev/null; then
|
||||||
|
echo "Please install homebrew first. This is a requirement for the remaining setup."
|
||||||
|
echo "You can find that here: https://brew.sh"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Install base python packages
|
||||||
|
echo "Installing Python 3.10 if not found."
|
||||||
|
brew ls --versions python@3.10 >/dev/null || brew install python@3.10
|
||||||
|
echo "Installing Python-TK 3.10 if not found."
|
||||||
|
brew ls --versions python-tk@3.10 >/dev/null || brew install python-tk@3.10
|
||||||
|
|
||||||
|
if command -v python3.10 >/dev/null; then
|
||||||
|
python3.10 -m venv venv
|
||||||
|
source venv/bin/activate
|
||||||
|
|
||||||
|
# DEBUG ONLY
|
||||||
|
#pip install pydevd-pycharm~=223.8836.43
|
||||||
|
|
||||||
|
# Tensorflow installation
|
||||||
|
if wget https://github.com/apple/tensorflow_macos/releases/download/v0.1alpha3/tensorflow_macos-0.1a3-cp38-cp38-macosx_11_0_arm64.whl /tmp; then
|
||||||
|
python -m pip install tensorflow==0.1a3 -f https://github.com/apple/tensorflow_macos/releases/download/v0.1alpha3/tensorflow_macos-0.1a3-cp38-cp38-macosx_11_0_arm64.whl
|
||||||
|
rm -f /tmp/tensorflow_macos-0.1a3-cp38-cp38-macosx_11_0_arm64.whl
|
||||||
|
fi
|
||||||
|
|
||||||
|
pip install torch==2.0.0 torchvision==0.15.1 -f https://download.pytorch.org/whl/cpu/torch_stable.html
|
||||||
|
python -m pip install --use-pep517 --upgrade -r requirements_macos.txt
|
||||||
|
accelerate config
|
||||||
|
echo -e "Setup finished! Run ./gui.sh to start."
|
||||||
|
else
|
||||||
|
echo "Python not found. Please ensure you install Python."
|
||||||
|
echo "The brew command for Python 3.10 is: brew install python@3.10"
|
||||||
|
exit 1
|
||||||
|
fi
|
@ -570,7 +570,10 @@ def ti_tab(
|
|||||||
v_parameterization,
|
v_parameterization,
|
||||||
save_model_as,
|
save_model_as,
|
||||||
model_list,
|
model_list,
|
||||||
) = gradio_source_model()
|
) = gradio_source_model(save_model_as_choices = [
|
||||||
|
'ckpt',
|
||||||
|
'safetensors',
|
||||||
|
])
|
||||||
|
|
||||||
with gr.Tab('Folders'):
|
with gr.Tab('Folders'):
|
||||||
with gr.Row():
|
with gr.Row():
|
||||||
|
@ -1,11 +1,17 @@
|
|||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
import pkg_resources
|
import pkg_resources
|
||||||
|
import argparse
|
||||||
|
|
||||||
|
# Parse command line arguments
|
||||||
|
parser = argparse.ArgumentParser(description="Validate that requirements are satisfied.")
|
||||||
|
parser.add_argument('-r', '--requirements', type=str, default='requirements.txt', help="Path to the requirements file.")
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
print("Validating that requirements are satisfied.")
|
print("Validating that requirements are satisfied.")
|
||||||
|
|
||||||
# Load the requirements from the requirements.txt file
|
# Load the requirements from the specified requirements file
|
||||||
with open('requirements.txt') as f:
|
with open(args.requirements) as f:
|
||||||
requirements = f.readlines()
|
requirements = f.readlines()
|
||||||
|
|
||||||
# Check each requirement against the installed packages
|
# Check each requirement against the installed packages
|
||||||
@ -34,7 +40,7 @@ if missing_requirements or wrong_version_requirements:
|
|||||||
for requirement, expected_version, actual_version in wrong_version_requirements:
|
for requirement, expected_version, actual_version in wrong_version_requirements:
|
||||||
print(f" - {requirement} (expected version {expected_version}, found version {actual_version})")
|
print(f" - {requirement} (expected version {expected_version}, found version {actual_version})")
|
||||||
upgrade_script = "upgrade.ps1" if os.name == "nt" else "upgrade.sh"
|
upgrade_script = "upgrade.ps1" if os.name == "nt" else "upgrade.sh"
|
||||||
print(f"\nRun \033[33m{upgrade_script}\033[0m or \033[33mpip install -U -r requirements.txt\033[0m to resolve the missing requirements listed above...")
|
print(f"\nRun \033[33m{upgrade_script}\033[0m or \033[33mpip install -U -r {args.requirements}\033[0m to resolve the missing requirements listed above...")
|
||||||
|
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
|
16
upgrade_macos.sh
Executable file
16
upgrade_macos.sh
Executable file
@ -0,0 +1,16 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Check if there are any changes that need to be committed
|
||||||
|
if [[ -n $(git status --short) ]]; then
|
||||||
|
echo "There are changes that need to be committed. Please stash or undo your changes before running this script." >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Pull the latest changes from the remote repository
|
||||||
|
git pull
|
||||||
|
|
||||||
|
# Activate the virtual environment
|
||||||
|
source venv/bin/activate
|
||||||
|
|
||||||
|
# Upgrade the required packages
|
||||||
|
pip install --upgrade -r requirements_macos.txt
|
Loading…
Reference in New Issue
Block a user