Merge dreambooth and Finetune is a common GUI
Merge dreambooth and Finetune is a common GUI
This commit is contained in:
commit
89d275dff7
@ -14,6 +14,9 @@ You can find the finetune solution spercific [Finetune README](README_finetune.m
|
|||||||
|
|
||||||
## Change history
|
## Change history
|
||||||
|
|
||||||
|
* 12/22 (v18.7) update:
|
||||||
|
- Merge dreambooth and finetune is a common GUI
|
||||||
|
- General bug fixes and code improvements
|
||||||
* 12/21 (v18.6.1) update:
|
* 12/21 (v18.6.1) update:
|
||||||
- fix issue with dataset balancing when the number of detected images in the folder is 0
|
- fix issue with dataset balancing when the number of detected images in the folder is 0
|
||||||
|
|
||||||
|
@ -71,7 +71,7 @@ When a new release comes out you can upgrade your repo with the following comman
|
|||||||
.\upgrade.bat
|
.\upgrade.bat
|
||||||
```
|
```
|
||||||
|
|
||||||
or you can do it manually with
|
alternatively you can do it manually with
|
||||||
|
|
||||||
```powershell
|
```powershell
|
||||||
cd kohya_ss
|
cd kohya_ss
|
||||||
@ -84,15 +84,27 @@ Once the commands have completed successfully you should be ready to use the new
|
|||||||
|
|
||||||
## GUI
|
## GUI
|
||||||
|
|
||||||
There is now support for GUI based training using gradio. You can start the GUI interface by running:
|
There is now support for GUI based training using gradio. You can start the complete kohya training GUI interface by running:
|
||||||
|
|
||||||
```powershell
|
```powershell
|
||||||
.\dreambooth.bat
|
.\kohya.cmd
|
||||||
|
```
|
||||||
|
|
||||||
|
and select the Dreambooth tab.
|
||||||
|
|
||||||
|
Alternativelly you can use the Dreambooth focus GUI with
|
||||||
|
|
||||||
|
```powershell
|
||||||
|
.\dreambooth.cmd
|
||||||
```
|
```
|
||||||
|
|
||||||
## CLI
|
## CLI
|
||||||
|
|
||||||
You can find various examples of how to leverage the fine_tune.py in this folder: https://github.com/bmaltais/kohya_ss/tree/master/examples
|
You can find various examples of how to leverage the `train_db.py` in this folder: https://github.com/bmaltais/kohya_ss/tree/master/examples
|
||||||
|
|
||||||
|
## Support
|
||||||
|
|
||||||
|
Drop by the discord server for support: https://discord.com/channels/1041518562487058594/1041518563242020906
|
||||||
|
|
||||||
## Quickstart screencast
|
## Quickstart screencast
|
||||||
|
|
||||||
|
@ -107,15 +107,23 @@ You can also use the `Captioning` tool found under the `Utilities` tab in the GU
|
|||||||
|
|
||||||
## GUI
|
## GUI
|
||||||
|
|
||||||
Support for GUI based training using gradio. You can start the GUI interface by running:
|
There is now support for GUI based training using gradio. You can start the complete kohya training GUI interface by running:
|
||||||
|
|
||||||
```powershell
|
```powershell
|
||||||
.\finetune.bat
|
.\kohya.cmd
|
||||||
|
```
|
||||||
|
|
||||||
|
and select the Finetune tab.
|
||||||
|
|
||||||
|
Alternativelly you can use the Finetune focus GUI with
|
||||||
|
|
||||||
|
```powershell
|
||||||
|
.\finetune.cmd
|
||||||
```
|
```
|
||||||
|
|
||||||
## CLI
|
## CLI
|
||||||
|
|
||||||
You can find various examples of how to leverage the fine_tune.py in this folder: https://github.com/bmaltais/kohya_ss/tree/master/examples
|
You can find various examples of how to leverage the `fine_tune.py` in this folder: https://github.com/bmaltais/kohya_ss/tree/master/examples
|
||||||
|
|
||||||
## Support
|
## Support
|
||||||
|
|
||||||
|
File diff suppressed because it is too large
Load Diff
2115
fine_tune.py
2115
fine_tune.py
File diff suppressed because it is too large
Load Diff
909
finetune_gui.py
909
finetune_gui.py
@ -6,17 +6,12 @@ import subprocess
|
|||||||
import pathlib
|
import pathlib
|
||||||
import shutil
|
import shutil
|
||||||
import argparse
|
import argparse
|
||||||
|
|
||||||
# from easygui import fileopenbox, filesavebox, diropenbox, msgbox
|
|
||||||
from library.basic_caption_gui import gradio_basic_caption_gui_tab
|
|
||||||
from library.convert_model_gui import gradio_convert_model_tab
|
|
||||||
from library.blip_caption_gui import gradio_blip_caption_gui_tab
|
|
||||||
from library.wd14_caption_gui import gradio_wd14_caption_gui_tab
|
|
||||||
from library.common_gui import (
|
from library.common_gui import (
|
||||||
get_folder_path,
|
get_folder_path,
|
||||||
get_file_path,
|
get_file_path,
|
||||||
get_saveasfile_path,
|
get_saveasfile_path,
|
||||||
)
|
)
|
||||||
|
from library.utilities import utilities_tab
|
||||||
|
|
||||||
folder_symbol = '\U0001f4c2' # 📂
|
folder_symbol = '\U0001f4c2' # 📂
|
||||||
refresh_symbol = '\U0001f504' # 🔄
|
refresh_symbol = '\U0001f504' # 🔄
|
||||||
@ -49,7 +44,6 @@ def save_configuration(
|
|||||||
train_text_encoder,
|
train_text_encoder,
|
||||||
create_buckets,
|
create_buckets,
|
||||||
create_caption,
|
create_caption,
|
||||||
train,
|
|
||||||
save_model_as,
|
save_model_as,
|
||||||
caption_extension,
|
caption_extension,
|
||||||
):
|
):
|
||||||
@ -94,7 +88,6 @@ def save_configuration(
|
|||||||
'train_text_encoder': train_text_encoder,
|
'train_text_encoder': train_text_encoder,
|
||||||
'create_buckets': create_buckets,
|
'create_buckets': create_buckets,
|
||||||
'create_caption': create_caption,
|
'create_caption': create_caption,
|
||||||
'train': train,
|
|
||||||
'save_model_as': save_model_as,
|
'save_model_as': save_model_as,
|
||||||
'caption_extension': caption_extension,
|
'caption_extension': caption_extension,
|
||||||
}
|
}
|
||||||
@ -130,7 +123,6 @@ def open_config_file(
|
|||||||
train_text_encoder,
|
train_text_encoder,
|
||||||
create_buckets,
|
create_buckets,
|
||||||
create_caption,
|
create_caption,
|
||||||
train,
|
|
||||||
save_model_as,
|
save_model_as,
|
||||||
caption_extension,
|
caption_extension,
|
||||||
):
|
):
|
||||||
@ -175,7 +167,6 @@ def open_config_file(
|
|||||||
my_data.get('train_text_encoder', train_text_encoder),
|
my_data.get('train_text_encoder', train_text_encoder),
|
||||||
my_data.get('create_buckets', create_buckets),
|
my_data.get('create_buckets', create_buckets),
|
||||||
my_data.get('create_caption', create_caption),
|
my_data.get('create_caption', create_caption),
|
||||||
my_data.get('train', train),
|
|
||||||
my_data.get('save_model_as', save_model_as),
|
my_data.get('save_model_as', save_model_as),
|
||||||
my_data.get('caption_extension', caption_extension),
|
my_data.get('caption_extension', caption_extension),
|
||||||
)
|
)
|
||||||
@ -184,7 +175,6 @@ def open_config_file(
|
|||||||
def train_model(
|
def train_model(
|
||||||
generate_caption_database,
|
generate_caption_database,
|
||||||
generate_image_buckets,
|
generate_image_buckets,
|
||||||
train,
|
|
||||||
pretrained_model_name_or_path,
|
pretrained_model_name_or_path,
|
||||||
v2,
|
v2,
|
||||||
v_parameterization,
|
v_parameterization,
|
||||||
@ -267,59 +257,58 @@ def train_model(
|
|||||||
# Run the command
|
# Run the command
|
||||||
subprocess.run(command)
|
subprocess.run(command)
|
||||||
|
|
||||||
if train:
|
image_num = len(
|
||||||
image_num = len(
|
[f for f in os.listdir(image_folder) if f.endswith('.npz')]
|
||||||
[f for f in os.listdir(image_folder) if f.endswith('.npz')]
|
)
|
||||||
)
|
print(f'image_num = {image_num}')
|
||||||
print(f'image_num = {image_num}')
|
|
||||||
|
|
||||||
repeats = int(image_num) * int(dataset_repeats)
|
repeats = int(image_num) * int(dataset_repeats)
|
||||||
print(f'repeats = {str(repeats)}')
|
print(f'repeats = {str(repeats)}')
|
||||||
|
|
||||||
# calculate max_train_steps
|
# calculate max_train_steps
|
||||||
max_train_steps = int(
|
max_train_steps = int(
|
||||||
math.ceil(float(repeats) / int(train_batch_size) * int(epoch))
|
math.ceil(float(repeats) / int(train_batch_size) * int(epoch))
|
||||||
)
|
)
|
||||||
print(f'max_train_steps = {max_train_steps}')
|
print(f'max_train_steps = {max_train_steps}')
|
||||||
|
|
||||||
lr_warmup_steps = round(
|
lr_warmup_steps = round(
|
||||||
float(int(lr_warmup) * int(max_train_steps) / 100)
|
float(int(lr_warmup) * int(max_train_steps) / 100)
|
||||||
)
|
)
|
||||||
print(f'lr_warmup_steps = {lr_warmup_steps}')
|
print(f'lr_warmup_steps = {lr_warmup_steps}')
|
||||||
|
|
||||||
run_cmd = f'accelerate launch --num_cpu_threads_per_process={num_cpu_threads_per_process} "./fine_tune.py"'
|
run_cmd = f'accelerate launch --num_cpu_threads_per_process={num_cpu_threads_per_process} "./fine_tune.py"'
|
||||||
if v2:
|
if v2:
|
||||||
run_cmd += ' --v2'
|
run_cmd += ' --v2'
|
||||||
if v_parameterization:
|
if v_parameterization:
|
||||||
run_cmd += ' --v_parameterization'
|
run_cmd += ' --v_parameterization'
|
||||||
if train_text_encoder:
|
if train_text_encoder:
|
||||||
run_cmd += ' --train_text_encoder'
|
run_cmd += ' --train_text_encoder'
|
||||||
run_cmd += (
|
run_cmd += (
|
||||||
f' --pretrained_model_name_or_path={pretrained_model_name_or_path}'
|
f' --pretrained_model_name_or_path={pretrained_model_name_or_path}'
|
||||||
)
|
)
|
||||||
run_cmd += f' --in_json={train_dir}/meta_lat.json'
|
run_cmd += f' --in_json={train_dir}/meta_lat.json'
|
||||||
run_cmd += f' --train_data_dir={image_folder}'
|
run_cmd += f' --train_data_dir={image_folder}'
|
||||||
run_cmd += f' --output_dir={output_dir}'
|
run_cmd += f' --output_dir={output_dir}'
|
||||||
if not logging_dir == '':
|
if not logging_dir == '':
|
||||||
run_cmd += f' --logging_dir={logging_dir}'
|
run_cmd += f' --logging_dir={logging_dir}'
|
||||||
run_cmd += f' --train_batch_size={train_batch_size}'
|
run_cmd += f' --train_batch_size={train_batch_size}'
|
||||||
run_cmd += f' --dataset_repeats={dataset_repeats}'
|
run_cmd += f' --dataset_repeats={dataset_repeats}'
|
||||||
run_cmd += f' --learning_rate={learning_rate}'
|
run_cmd += f' --learning_rate={learning_rate}'
|
||||||
run_cmd += f' --lr_scheduler={lr_scheduler}'
|
run_cmd += f' --lr_scheduler={lr_scheduler}'
|
||||||
run_cmd += f' --lr_warmup_steps={lr_warmup_steps}'
|
run_cmd += f' --lr_warmup_steps={lr_warmup_steps}'
|
||||||
run_cmd += f' --max_train_steps={max_train_steps}'
|
run_cmd += f' --max_train_steps={max_train_steps}'
|
||||||
run_cmd += f' --use_8bit_adam'
|
run_cmd += f' --use_8bit_adam'
|
||||||
run_cmd += f' --xformers'
|
run_cmd += f' --xformers'
|
||||||
run_cmd += f' --mixed_precision={mixed_precision}'
|
run_cmd += f' --mixed_precision={mixed_precision}'
|
||||||
run_cmd += f' --save_every_n_epochs={save_every_n_epochs}'
|
run_cmd += f' --save_every_n_epochs={save_every_n_epochs}'
|
||||||
run_cmd += f' --seed={seed}'
|
run_cmd += f' --seed={seed}'
|
||||||
run_cmd += f' --save_precision={save_precision}'
|
run_cmd += f' --save_precision={save_precision}'
|
||||||
if not save_model_as == 'same as source model':
|
if not save_model_as == 'same as source model':
|
||||||
run_cmd += f' --save_model_as={save_model_as}'
|
run_cmd += f' --save_model_as={save_model_as}'
|
||||||
|
|
||||||
print(run_cmd)
|
print(run_cmd)
|
||||||
# Run the command
|
# Run the command
|
||||||
subprocess.run(run_cmd)
|
subprocess.run(run_cmd)
|
||||||
|
|
||||||
# check if output_dir/last is a folder... therefore it is a diffuser model
|
# check if output_dir/last is a folder... therefore it is a diffuser model
|
||||||
last_dir = pathlib.Path(f'{output_dir}/last')
|
last_dir = pathlib.Path(f'{output_dir}/last')
|
||||||
@ -386,6 +375,7 @@ def remove_doublequote(file_path):
|
|||||||
|
|
||||||
return file_path
|
return file_path
|
||||||
|
|
||||||
|
|
||||||
def UI(username, password):
|
def UI(username, password):
|
||||||
|
|
||||||
css = ''
|
css = ''
|
||||||
@ -398,399 +388,10 @@ def UI(username, password):
|
|||||||
interface = gr.Blocks(css=css)
|
interface = gr.Blocks(css=css)
|
||||||
|
|
||||||
with interface:
|
with interface:
|
||||||
dummy_true = gr.Label(value=True, visible=False)
|
with gr.Tab("Finetune"):
|
||||||
dummy_false = gr.Label(value=False, visible=False)
|
finetune_tab()
|
||||||
with gr.Tab('Finetuning'):
|
with gr.Tab("Utilities"):
|
||||||
gr.Markdown('Enter kohya finetuner parameter using this interface.')
|
utilities_tab(enable_dreambooth_tab=False)
|
||||||
with gr.Accordion('Configuration File Load/Save', open=False):
|
|
||||||
with gr.Row():
|
|
||||||
button_open_config = gr.Button(
|
|
||||||
f'Open {folder_symbol}', elem_id='open_folder'
|
|
||||||
)
|
|
||||||
button_save_config = gr.Button(
|
|
||||||
f'Save {save_style_symbol}', elem_id='open_folder'
|
|
||||||
)
|
|
||||||
button_save_as_config = gr.Button(
|
|
||||||
f'Save as... {save_style_symbol}', elem_id='open_folder'
|
|
||||||
)
|
|
||||||
config_file_name = gr.Textbox(
|
|
||||||
label='', placeholder='type file path or use buttons...'
|
|
||||||
)
|
|
||||||
config_file_name.change(
|
|
||||||
remove_doublequote,
|
|
||||||
inputs=[config_file_name],
|
|
||||||
outputs=[config_file_name],
|
|
||||||
)
|
|
||||||
with gr.Tab('Source model'):
|
|
||||||
# Define the input elements
|
|
||||||
with gr.Row():
|
|
||||||
pretrained_model_name_or_path_input = gr.Textbox(
|
|
||||||
label='Pretrained model name or path',
|
|
||||||
placeholder='enter the path to custom model or name of pretrained model',
|
|
||||||
)
|
|
||||||
pretrained_model_name_or_path_file = gr.Button(
|
|
||||||
document_symbol, elem_id='open_folder_small'
|
|
||||||
)
|
|
||||||
pretrained_model_name_or_path_file.click(
|
|
||||||
get_file_path,
|
|
||||||
inputs=pretrained_model_name_or_path_input,
|
|
||||||
outputs=pretrained_model_name_or_path_input,
|
|
||||||
)
|
|
||||||
pretrained_model_name_or_path_folder = gr.Button(
|
|
||||||
folder_symbol, elem_id='open_folder_small'
|
|
||||||
)
|
|
||||||
pretrained_model_name_or_path_folder.click(
|
|
||||||
get_folder_path,
|
|
||||||
inputs=pretrained_model_name_or_path_input,
|
|
||||||
outputs=pretrained_model_name_or_path_input,
|
|
||||||
)
|
|
||||||
model_list = gr.Dropdown(
|
|
||||||
label='(Optional) Model Quick Pick',
|
|
||||||
choices=[
|
|
||||||
'custom',
|
|
||||||
'stabilityai/stable-diffusion-2-1-base',
|
|
||||||
'stabilityai/stable-diffusion-2-base',
|
|
||||||
'stabilityai/stable-diffusion-2-1',
|
|
||||||
'stabilityai/stable-diffusion-2',
|
|
||||||
'runwayml/stable-diffusion-v1-5',
|
|
||||||
'CompVis/stable-diffusion-v1-4',
|
|
||||||
],
|
|
||||||
)
|
|
||||||
save_model_as_dropdown = gr.Dropdown(
|
|
||||||
label='Save trained model as',
|
|
||||||
choices=[
|
|
||||||
'same as source model',
|
|
||||||
'ckpt',
|
|
||||||
'diffusers',
|
|
||||||
'diffusers_safetensors',
|
|
||||||
'safetensors',
|
|
||||||
],
|
|
||||||
value='same as source model',
|
|
||||||
)
|
|
||||||
|
|
||||||
with gr.Row():
|
|
||||||
v2_input = gr.Checkbox(label='v2', value=True)
|
|
||||||
v_parameterization_input = gr.Checkbox(
|
|
||||||
label='v_parameterization', value=False
|
|
||||||
)
|
|
||||||
model_list.change(
|
|
||||||
set_pretrained_model_name_or_path_input,
|
|
||||||
inputs=[model_list, v2_input, v_parameterization_input],
|
|
||||||
outputs=[
|
|
||||||
pretrained_model_name_or_path_input,
|
|
||||||
v2_input,
|
|
||||||
v_parameterization_input,
|
|
||||||
],
|
|
||||||
)
|
|
||||||
with gr.Tab('Directories'):
|
|
||||||
with gr.Row():
|
|
||||||
train_dir_input = gr.Textbox(
|
|
||||||
label='Training config folder',
|
|
||||||
placeholder='folder where the training configuration files will be saved',
|
|
||||||
)
|
|
||||||
train_dir_folder = gr.Button(
|
|
||||||
folder_symbol, elem_id='open_folder_small'
|
|
||||||
)
|
|
||||||
train_dir_folder.click(
|
|
||||||
get_folder_path, outputs=train_dir_input
|
|
||||||
)
|
|
||||||
|
|
||||||
image_folder_input = gr.Textbox(
|
|
||||||
label='Training Image folder',
|
|
||||||
placeholder='folder where the training images are located',
|
|
||||||
)
|
|
||||||
image_folder_input_folder = gr.Button(
|
|
||||||
folder_symbol, elem_id='open_folder_small'
|
|
||||||
)
|
|
||||||
image_folder_input_folder.click(
|
|
||||||
get_folder_path, outputs=image_folder_input
|
|
||||||
)
|
|
||||||
with gr.Row():
|
|
||||||
output_dir_input = gr.Textbox(
|
|
||||||
label='Output folder',
|
|
||||||
placeholder='folder where the model will be saved',
|
|
||||||
)
|
|
||||||
output_dir_input_folder = gr.Button(
|
|
||||||
folder_symbol, elem_id='open_folder_small'
|
|
||||||
)
|
|
||||||
output_dir_input_folder.click(
|
|
||||||
get_folder_path, outputs=output_dir_input
|
|
||||||
)
|
|
||||||
|
|
||||||
logging_dir_input = gr.Textbox(
|
|
||||||
label='Logging folder',
|
|
||||||
placeholder='Optional: enable logging and output TensorBoard log to this folder',
|
|
||||||
)
|
|
||||||
logging_dir_input_folder = gr.Button(
|
|
||||||
folder_symbol, elem_id='open_folder_small'
|
|
||||||
)
|
|
||||||
logging_dir_input_folder.click(
|
|
||||||
get_folder_path, outputs=logging_dir_input
|
|
||||||
)
|
|
||||||
train_dir_input.change(
|
|
||||||
remove_doublequote,
|
|
||||||
inputs=[train_dir_input],
|
|
||||||
outputs=[train_dir_input],
|
|
||||||
)
|
|
||||||
image_folder_input.change(
|
|
||||||
remove_doublequote,
|
|
||||||
inputs=[image_folder_input],
|
|
||||||
outputs=[image_folder_input],
|
|
||||||
)
|
|
||||||
output_dir_input.change(
|
|
||||||
remove_doublequote,
|
|
||||||
inputs=[output_dir_input],
|
|
||||||
outputs=[output_dir_input],
|
|
||||||
)
|
|
||||||
with gr.Tab('Training parameters'):
|
|
||||||
with gr.Row():
|
|
||||||
learning_rate_input = gr.Textbox(
|
|
||||||
label='Learning rate', value=1e-6
|
|
||||||
)
|
|
||||||
lr_scheduler_input = gr.Dropdown(
|
|
||||||
label='LR Scheduler',
|
|
||||||
choices=[
|
|
||||||
'constant',
|
|
||||||
'constant_with_warmup',
|
|
||||||
'cosine',
|
|
||||||
'cosine_with_restarts',
|
|
||||||
'linear',
|
|
||||||
'polynomial',
|
|
||||||
],
|
|
||||||
value='constant',
|
|
||||||
)
|
|
||||||
lr_warmup_input = gr.Textbox(label='LR warmup', value=0)
|
|
||||||
with gr.Row():
|
|
||||||
dataset_repeats_input = gr.Textbox(
|
|
||||||
label='Dataset repeats', value=40
|
|
||||||
)
|
|
||||||
train_batch_size_input = gr.Slider(
|
|
||||||
minimum=1,
|
|
||||||
maximum=32,
|
|
||||||
label='Train batch size',
|
|
||||||
value=1,
|
|
||||||
step=1,
|
|
||||||
)
|
|
||||||
epoch_input = gr.Textbox(label='Epoch', value=1)
|
|
||||||
save_every_n_epochs_input = gr.Textbox(
|
|
||||||
label='Save every N epochs', value=1
|
|
||||||
)
|
|
||||||
with gr.Row():
|
|
||||||
mixed_precision_input = gr.Dropdown(
|
|
||||||
label='Mixed precision',
|
|
||||||
choices=[
|
|
||||||
'no',
|
|
||||||
'fp16',
|
|
||||||
'bf16',
|
|
||||||
],
|
|
||||||
value='fp16',
|
|
||||||
)
|
|
||||||
save_precision_input = gr.Dropdown(
|
|
||||||
label='Save precision',
|
|
||||||
choices=[
|
|
||||||
'float',
|
|
||||||
'fp16',
|
|
||||||
'bf16',
|
|
||||||
],
|
|
||||||
value='fp16',
|
|
||||||
)
|
|
||||||
num_cpu_threads_per_process_input = gr.Slider(
|
|
||||||
minimum=1,
|
|
||||||
maximum=os.cpu_count(),
|
|
||||||
step=1,
|
|
||||||
label='Number of CPU threads per process',
|
|
||||||
value=os.cpu_count(),
|
|
||||||
)
|
|
||||||
with gr.Row():
|
|
||||||
seed_input = gr.Textbox(label='Seed', value=1234)
|
|
||||||
max_resolution_input = gr.Textbox(
|
|
||||||
label='Max resolution', value='512,512'
|
|
||||||
)
|
|
||||||
with gr.Row():
|
|
||||||
caption_extention_input = gr.Textbox(
|
|
||||||
label='Caption Extension',
|
|
||||||
placeholder='(Optional) Extension for caption files. default: .txt',
|
|
||||||
)
|
|
||||||
train_text_encoder_input = gr.Checkbox(
|
|
||||||
label='Train text encoder', value=True
|
|
||||||
)
|
|
||||||
with gr.Box():
|
|
||||||
with gr.Row():
|
|
||||||
create_caption = gr.Checkbox(
|
|
||||||
label='Generate caption database', value=True
|
|
||||||
)
|
|
||||||
create_buckets = gr.Checkbox(
|
|
||||||
label='Generate image buckets', value=True
|
|
||||||
)
|
|
||||||
train = gr.Checkbox(label='Train model', value=True)
|
|
||||||
|
|
||||||
button_run = gr.Button('Run')
|
|
||||||
|
|
||||||
button_run.click(
|
|
||||||
train_model,
|
|
||||||
inputs=[
|
|
||||||
create_caption,
|
|
||||||
create_buckets,
|
|
||||||
train,
|
|
||||||
pretrained_model_name_or_path_input,
|
|
||||||
v2_input,
|
|
||||||
v_parameterization_input,
|
|
||||||
train_dir_input,
|
|
||||||
image_folder_input,
|
|
||||||
output_dir_input,
|
|
||||||
logging_dir_input,
|
|
||||||
max_resolution_input,
|
|
||||||
learning_rate_input,
|
|
||||||
lr_scheduler_input,
|
|
||||||
lr_warmup_input,
|
|
||||||
dataset_repeats_input,
|
|
||||||
train_batch_size_input,
|
|
||||||
epoch_input,
|
|
||||||
save_every_n_epochs_input,
|
|
||||||
mixed_precision_input,
|
|
||||||
save_precision_input,
|
|
||||||
seed_input,
|
|
||||||
num_cpu_threads_per_process_input,
|
|
||||||
train_text_encoder_input,
|
|
||||||
save_model_as_dropdown,
|
|
||||||
caption_extention_input,
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
button_open_config.click(
|
|
||||||
open_config_file,
|
|
||||||
inputs=[
|
|
||||||
config_file_name,
|
|
||||||
pretrained_model_name_or_path_input,
|
|
||||||
v2_input,
|
|
||||||
v_parameterization_input,
|
|
||||||
train_dir_input,
|
|
||||||
image_folder_input,
|
|
||||||
output_dir_input,
|
|
||||||
logging_dir_input,
|
|
||||||
max_resolution_input,
|
|
||||||
learning_rate_input,
|
|
||||||
lr_scheduler_input,
|
|
||||||
lr_warmup_input,
|
|
||||||
dataset_repeats_input,
|
|
||||||
train_batch_size_input,
|
|
||||||
epoch_input,
|
|
||||||
save_every_n_epochs_input,
|
|
||||||
mixed_precision_input,
|
|
||||||
save_precision_input,
|
|
||||||
seed_input,
|
|
||||||
num_cpu_threads_per_process_input,
|
|
||||||
train_text_encoder_input,
|
|
||||||
create_buckets,
|
|
||||||
create_caption,
|
|
||||||
train,
|
|
||||||
save_model_as_dropdown,
|
|
||||||
caption_extention_input,
|
|
||||||
],
|
|
||||||
outputs=[
|
|
||||||
config_file_name,
|
|
||||||
pretrained_model_name_or_path_input,
|
|
||||||
v2_input,
|
|
||||||
v_parameterization_input,
|
|
||||||
train_dir_input,
|
|
||||||
image_folder_input,
|
|
||||||
output_dir_input,
|
|
||||||
logging_dir_input,
|
|
||||||
max_resolution_input,
|
|
||||||
learning_rate_input,
|
|
||||||
lr_scheduler_input,
|
|
||||||
lr_warmup_input,
|
|
||||||
dataset_repeats_input,
|
|
||||||
train_batch_size_input,
|
|
||||||
epoch_input,
|
|
||||||
save_every_n_epochs_input,
|
|
||||||
mixed_precision_input,
|
|
||||||
save_precision_input,
|
|
||||||
seed_input,
|
|
||||||
num_cpu_threads_per_process_input,
|
|
||||||
train_text_encoder_input,
|
|
||||||
create_buckets,
|
|
||||||
create_caption,
|
|
||||||
train,
|
|
||||||
save_model_as_dropdown,
|
|
||||||
caption_extention_input,
|
|
||||||
],
|
|
||||||
)
|
|
||||||
|
|
||||||
button_save_config.click(
|
|
||||||
save_configuration,
|
|
||||||
inputs=[
|
|
||||||
dummy_false,
|
|
||||||
config_file_name,
|
|
||||||
pretrained_model_name_or_path_input,
|
|
||||||
v2_input,
|
|
||||||
v_parameterization_input,
|
|
||||||
train_dir_input,
|
|
||||||
image_folder_input,
|
|
||||||
output_dir_input,
|
|
||||||
logging_dir_input,
|
|
||||||
max_resolution_input,
|
|
||||||
learning_rate_input,
|
|
||||||
lr_scheduler_input,
|
|
||||||
lr_warmup_input,
|
|
||||||
dataset_repeats_input,
|
|
||||||
train_batch_size_input,
|
|
||||||
epoch_input,
|
|
||||||
save_every_n_epochs_input,
|
|
||||||
mixed_precision_input,
|
|
||||||
save_precision_input,
|
|
||||||
seed_input,
|
|
||||||
num_cpu_threads_per_process_input,
|
|
||||||
train_text_encoder_input,
|
|
||||||
create_buckets,
|
|
||||||
create_caption,
|
|
||||||
train,
|
|
||||||
save_model_as_dropdown,
|
|
||||||
caption_extention_input,
|
|
||||||
],
|
|
||||||
outputs=[config_file_name],
|
|
||||||
)
|
|
||||||
|
|
||||||
button_save_as_config.click(
|
|
||||||
save_configuration,
|
|
||||||
inputs=[
|
|
||||||
dummy_true,
|
|
||||||
config_file_name,
|
|
||||||
pretrained_model_name_or_path_input,
|
|
||||||
v2_input,
|
|
||||||
v_parameterization_input,
|
|
||||||
train_dir_input,
|
|
||||||
image_folder_input,
|
|
||||||
output_dir_input,
|
|
||||||
logging_dir_input,
|
|
||||||
max_resolution_input,
|
|
||||||
learning_rate_input,
|
|
||||||
lr_scheduler_input,
|
|
||||||
lr_warmup_input,
|
|
||||||
dataset_repeats_input,
|
|
||||||
train_batch_size_input,
|
|
||||||
epoch_input,
|
|
||||||
save_every_n_epochs_input,
|
|
||||||
mixed_precision_input,
|
|
||||||
save_precision_input,
|
|
||||||
seed_input,
|
|
||||||
num_cpu_threads_per_process_input,
|
|
||||||
train_text_encoder_input,
|
|
||||||
create_buckets,
|
|
||||||
create_caption,
|
|
||||||
train,
|
|
||||||
save_model_as_dropdown,
|
|
||||||
caption_extention_input,
|
|
||||||
],
|
|
||||||
outputs=[config_file_name],
|
|
||||||
)
|
|
||||||
|
|
||||||
with gr.Tab('Utilities'):
|
|
||||||
gradio_basic_caption_gui_tab()
|
|
||||||
gradio_blip_caption_gui_tab()
|
|
||||||
gradio_wd14_caption_gui_tab()
|
|
||||||
gradio_convert_model_tab()
|
|
||||||
|
|
||||||
|
|
||||||
# Show the interface
|
# Show the interface
|
||||||
if not username == '':
|
if not username == '':
|
||||||
@ -798,13 +399,401 @@ def UI(username, password):
|
|||||||
else:
|
else:
|
||||||
interface.launch()
|
interface.launch()
|
||||||
|
|
||||||
|
def finetune_tab():
|
||||||
|
dummy_ft_true = gr.Label(value=True, visible=False)
|
||||||
|
dummy_ft_false = gr.Label(value=False, visible=False)
|
||||||
|
gr.Markdown(
|
||||||
|
'Enter kohya finetune training parameter using this interface.'
|
||||||
|
)
|
||||||
|
with gr.Accordion('Configuration File Load/Save', open=False):
|
||||||
|
with gr.Row():
|
||||||
|
button_open_config = gr.Button(
|
||||||
|
f'Open {folder_symbol}', elem_id='open_folder'
|
||||||
|
)
|
||||||
|
button_save_config = gr.Button(
|
||||||
|
f'Save {save_style_symbol}', elem_id='open_folder'
|
||||||
|
)
|
||||||
|
button_save_as_config = gr.Button(
|
||||||
|
f'Save as... {save_style_symbol}',
|
||||||
|
elem_id='open_folder',
|
||||||
|
)
|
||||||
|
config_file_name = gr.Textbox(
|
||||||
|
label='', placeholder='type file path or use buttons...'
|
||||||
|
)
|
||||||
|
config_file_name.change(
|
||||||
|
remove_doublequote,
|
||||||
|
inputs=[config_file_name],
|
||||||
|
outputs=[config_file_name],
|
||||||
|
)
|
||||||
|
with gr.Tab('Source model'):
|
||||||
|
# Define the input elements
|
||||||
|
with gr.Row():
|
||||||
|
pretrained_model_name_or_path_input = gr.Textbox(
|
||||||
|
label='Pretrained model name or path',
|
||||||
|
placeholder='enter the path to custom model or name of pretrained model',
|
||||||
|
)
|
||||||
|
pretrained_model_name_or_path_file = gr.Button(
|
||||||
|
document_symbol, elem_id='open_folder_small'
|
||||||
|
)
|
||||||
|
pretrained_model_name_or_path_file.click(
|
||||||
|
get_file_path,
|
||||||
|
inputs=pretrained_model_name_or_path_input,
|
||||||
|
outputs=pretrained_model_name_or_path_input,
|
||||||
|
)
|
||||||
|
pretrained_model_name_or_path_folder = gr.Button(
|
||||||
|
folder_symbol, elem_id='open_folder_small'
|
||||||
|
)
|
||||||
|
pretrained_model_name_or_path_folder.click(
|
||||||
|
get_folder_path,
|
||||||
|
inputs=pretrained_model_name_or_path_input,
|
||||||
|
outputs=pretrained_model_name_or_path_input,
|
||||||
|
)
|
||||||
|
model_list = gr.Dropdown(
|
||||||
|
label='(Optional) Model Quick Pick',
|
||||||
|
choices=[
|
||||||
|
'custom',
|
||||||
|
'stabilityai/stable-diffusion-2-1-base',
|
||||||
|
'stabilityai/stable-diffusion-2-base',
|
||||||
|
'stabilityai/stable-diffusion-2-1',
|
||||||
|
'stabilityai/stable-diffusion-2',
|
||||||
|
'runwayml/stable-diffusion-v1-5',
|
||||||
|
'CompVis/stable-diffusion-v1-4',
|
||||||
|
],
|
||||||
|
)
|
||||||
|
save_model_as_dropdown = gr.Dropdown(
|
||||||
|
label='Save trained model as',
|
||||||
|
choices=[
|
||||||
|
'same as source model',
|
||||||
|
'ckpt',
|
||||||
|
'diffusers',
|
||||||
|
'diffusers_safetensors',
|
||||||
|
'safetensors',
|
||||||
|
],
|
||||||
|
value='same as source model',
|
||||||
|
)
|
||||||
|
|
||||||
|
with gr.Row():
|
||||||
|
v2_input = gr.Checkbox(label='v2', value=True)
|
||||||
|
v_parameterization_input = gr.Checkbox(
|
||||||
|
label='v_parameterization', value=False
|
||||||
|
)
|
||||||
|
model_list.change(
|
||||||
|
set_pretrained_model_name_or_path_input,
|
||||||
|
inputs=[model_list, v2_input, v_parameterization_input],
|
||||||
|
outputs=[
|
||||||
|
pretrained_model_name_or_path_input,
|
||||||
|
v2_input,
|
||||||
|
v_parameterization_input,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
with gr.Tab('Directories'):
|
||||||
|
with gr.Row():
|
||||||
|
train_dir_input = gr.Textbox(
|
||||||
|
label='Training config folder',
|
||||||
|
placeholder='folder where the training configuration files will be saved',
|
||||||
|
)
|
||||||
|
train_dir_folder = gr.Button(
|
||||||
|
folder_symbol, elem_id='open_folder_small'
|
||||||
|
)
|
||||||
|
train_dir_folder.click(
|
||||||
|
get_folder_path, outputs=train_dir_input
|
||||||
|
)
|
||||||
|
|
||||||
|
image_folder_input = gr.Textbox(
|
||||||
|
label='Training Image folder',
|
||||||
|
placeholder='folder where the training images are located',
|
||||||
|
)
|
||||||
|
image_folder_input_folder = gr.Button(
|
||||||
|
folder_symbol, elem_id='open_folder_small'
|
||||||
|
)
|
||||||
|
image_folder_input_folder.click(
|
||||||
|
get_folder_path, outputs=image_folder_input
|
||||||
|
)
|
||||||
|
with gr.Row():
|
||||||
|
output_dir_input = gr.Textbox(
|
||||||
|
label='Output folder',
|
||||||
|
placeholder='folder where the model will be saved',
|
||||||
|
)
|
||||||
|
output_dir_input_folder = gr.Button(
|
||||||
|
folder_symbol, elem_id='open_folder_small'
|
||||||
|
)
|
||||||
|
output_dir_input_folder.click(
|
||||||
|
get_folder_path, outputs=output_dir_input
|
||||||
|
)
|
||||||
|
|
||||||
|
logging_dir_input = gr.Textbox(
|
||||||
|
label='Logging folder',
|
||||||
|
placeholder='Optional: enable logging and output TensorBoard log to this folder',
|
||||||
|
)
|
||||||
|
logging_dir_input_folder = gr.Button(
|
||||||
|
folder_symbol, elem_id='open_folder_small'
|
||||||
|
)
|
||||||
|
logging_dir_input_folder.click(
|
||||||
|
get_folder_path, outputs=logging_dir_input
|
||||||
|
)
|
||||||
|
train_dir_input.change(
|
||||||
|
remove_doublequote,
|
||||||
|
inputs=[train_dir_input],
|
||||||
|
outputs=[train_dir_input],
|
||||||
|
)
|
||||||
|
image_folder_input.change(
|
||||||
|
remove_doublequote,
|
||||||
|
inputs=[image_folder_input],
|
||||||
|
outputs=[image_folder_input],
|
||||||
|
)
|
||||||
|
output_dir_input.change(
|
||||||
|
remove_doublequote,
|
||||||
|
inputs=[output_dir_input],
|
||||||
|
outputs=[output_dir_input],
|
||||||
|
)
|
||||||
|
with gr.Tab('Training parameters'):
|
||||||
|
with gr.Row():
|
||||||
|
learning_rate_input = gr.Textbox(
|
||||||
|
label='Learning rate', value=1e-6
|
||||||
|
)
|
||||||
|
lr_scheduler_input = gr.Dropdown(
|
||||||
|
label='LR Scheduler',
|
||||||
|
choices=[
|
||||||
|
'constant',
|
||||||
|
'constant_with_warmup',
|
||||||
|
'cosine',
|
||||||
|
'cosine_with_restarts',
|
||||||
|
'linear',
|
||||||
|
'polynomial',
|
||||||
|
],
|
||||||
|
value='constant',
|
||||||
|
)
|
||||||
|
lr_warmup_input = gr.Textbox(label='LR warmup', value=0)
|
||||||
|
with gr.Row():
|
||||||
|
dataset_repeats_input = gr.Textbox(
|
||||||
|
label='Dataset repeats', value=40
|
||||||
|
)
|
||||||
|
train_batch_size_input = gr.Slider(
|
||||||
|
minimum=1,
|
||||||
|
maximum=32,
|
||||||
|
label='Train batch size',
|
||||||
|
value=1,
|
||||||
|
step=1,
|
||||||
|
)
|
||||||
|
epoch_input = gr.Textbox(label='Epoch', value=1)
|
||||||
|
save_every_n_epochs_input = gr.Textbox(
|
||||||
|
label='Save every N epochs', value=1
|
||||||
|
)
|
||||||
|
with gr.Row():
|
||||||
|
mixed_precision_input = gr.Dropdown(
|
||||||
|
label='Mixed precision',
|
||||||
|
choices=[
|
||||||
|
'no',
|
||||||
|
'fp16',
|
||||||
|
'bf16',
|
||||||
|
],
|
||||||
|
value='fp16',
|
||||||
|
)
|
||||||
|
save_precision_input = gr.Dropdown(
|
||||||
|
label='Save precision',
|
||||||
|
choices=[
|
||||||
|
'float',
|
||||||
|
'fp16',
|
||||||
|
'bf16',
|
||||||
|
],
|
||||||
|
value='fp16',
|
||||||
|
)
|
||||||
|
num_cpu_threads_per_process_input = gr.Slider(
|
||||||
|
minimum=1,
|
||||||
|
maximum=os.cpu_count(),
|
||||||
|
step=1,
|
||||||
|
label='Number of CPU threads per process',
|
||||||
|
value=os.cpu_count(),
|
||||||
|
)
|
||||||
|
with gr.Row():
|
||||||
|
seed_input = gr.Textbox(label='Seed', value=1234)
|
||||||
|
max_resolution_input = gr.Textbox(
|
||||||
|
label='Max resolution', value='512,512'
|
||||||
|
)
|
||||||
|
with gr.Row():
|
||||||
|
caption_extention_input = gr.Textbox(
|
||||||
|
label='Caption Extension',
|
||||||
|
placeholder='(Optional) Extension for caption files. default: .txt',
|
||||||
|
)
|
||||||
|
train_text_encoder_input = gr.Checkbox(
|
||||||
|
label='Train text encoder', value=True
|
||||||
|
)
|
||||||
|
with gr.Box():
|
||||||
|
with gr.Row():
|
||||||
|
create_caption = gr.Checkbox(
|
||||||
|
label='Generate caption database', value=True
|
||||||
|
)
|
||||||
|
create_buckets = gr.Checkbox(
|
||||||
|
label='Generate image buckets', value=True
|
||||||
|
)
|
||||||
|
|
||||||
|
button_run = gr.Button('Train model')
|
||||||
|
|
||||||
|
button_run.click(
|
||||||
|
train_model,
|
||||||
|
inputs=[
|
||||||
|
create_caption,
|
||||||
|
create_buckets,
|
||||||
|
pretrained_model_name_or_path_input,
|
||||||
|
v2_input,
|
||||||
|
v_parameterization_input,
|
||||||
|
train_dir_input,
|
||||||
|
image_folder_input,
|
||||||
|
output_dir_input,
|
||||||
|
logging_dir_input,
|
||||||
|
max_resolution_input,
|
||||||
|
learning_rate_input,
|
||||||
|
lr_scheduler_input,
|
||||||
|
lr_warmup_input,
|
||||||
|
dataset_repeats_input,
|
||||||
|
train_batch_size_input,
|
||||||
|
epoch_input,
|
||||||
|
save_every_n_epochs_input,
|
||||||
|
mixed_precision_input,
|
||||||
|
save_precision_input,
|
||||||
|
seed_input,
|
||||||
|
num_cpu_threads_per_process_input,
|
||||||
|
train_text_encoder_input,
|
||||||
|
save_model_as_dropdown,
|
||||||
|
caption_extention_input,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
button_open_config.click(
|
||||||
|
open_config_file,
|
||||||
|
inputs=[
|
||||||
|
config_file_name,
|
||||||
|
pretrained_model_name_or_path_input,
|
||||||
|
v2_input,
|
||||||
|
v_parameterization_input,
|
||||||
|
train_dir_input,
|
||||||
|
image_folder_input,
|
||||||
|
output_dir_input,
|
||||||
|
logging_dir_input,
|
||||||
|
max_resolution_input,
|
||||||
|
learning_rate_input,
|
||||||
|
lr_scheduler_input,
|
||||||
|
lr_warmup_input,
|
||||||
|
dataset_repeats_input,
|
||||||
|
train_batch_size_input,
|
||||||
|
epoch_input,
|
||||||
|
save_every_n_epochs_input,
|
||||||
|
mixed_precision_input,
|
||||||
|
save_precision_input,
|
||||||
|
seed_input,
|
||||||
|
num_cpu_threads_per_process_input,
|
||||||
|
train_text_encoder_input,
|
||||||
|
create_buckets,
|
||||||
|
create_caption,
|
||||||
|
save_model_as_dropdown,
|
||||||
|
caption_extention_input,
|
||||||
|
],
|
||||||
|
outputs=[
|
||||||
|
config_file_name,
|
||||||
|
pretrained_model_name_or_path_input,
|
||||||
|
v2_input,
|
||||||
|
v_parameterization_input,
|
||||||
|
train_dir_input,
|
||||||
|
image_folder_input,
|
||||||
|
output_dir_input,
|
||||||
|
logging_dir_input,
|
||||||
|
max_resolution_input,
|
||||||
|
learning_rate_input,
|
||||||
|
lr_scheduler_input,
|
||||||
|
lr_warmup_input,
|
||||||
|
dataset_repeats_input,
|
||||||
|
train_batch_size_input,
|
||||||
|
epoch_input,
|
||||||
|
save_every_n_epochs_input,
|
||||||
|
mixed_precision_input,
|
||||||
|
save_precision_input,
|
||||||
|
seed_input,
|
||||||
|
num_cpu_threads_per_process_input,
|
||||||
|
train_text_encoder_input,
|
||||||
|
create_buckets,
|
||||||
|
create_caption,
|
||||||
|
save_model_as_dropdown,
|
||||||
|
caption_extention_input,
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
button_save_config.click(
|
||||||
|
save_configuration,
|
||||||
|
inputs=[
|
||||||
|
dummy_ft_false,
|
||||||
|
config_file_name,
|
||||||
|
pretrained_model_name_or_path_input,
|
||||||
|
v2_input,
|
||||||
|
v_parameterization_input,
|
||||||
|
train_dir_input,
|
||||||
|
image_folder_input,
|
||||||
|
output_dir_input,
|
||||||
|
logging_dir_input,
|
||||||
|
max_resolution_input,
|
||||||
|
learning_rate_input,
|
||||||
|
lr_scheduler_input,
|
||||||
|
lr_warmup_input,
|
||||||
|
dataset_repeats_input,
|
||||||
|
train_batch_size_input,
|
||||||
|
epoch_input,
|
||||||
|
save_every_n_epochs_input,
|
||||||
|
mixed_precision_input,
|
||||||
|
save_precision_input,
|
||||||
|
seed_input,
|
||||||
|
num_cpu_threads_per_process_input,
|
||||||
|
train_text_encoder_input,
|
||||||
|
create_buckets,
|
||||||
|
create_caption,
|
||||||
|
save_model_as_dropdown,
|
||||||
|
caption_extention_input,
|
||||||
|
],
|
||||||
|
outputs=[config_file_name],
|
||||||
|
)
|
||||||
|
|
||||||
|
button_save_as_config.click(
|
||||||
|
save_configuration,
|
||||||
|
inputs=[
|
||||||
|
dummy_ft_true,
|
||||||
|
config_file_name,
|
||||||
|
pretrained_model_name_or_path_input,
|
||||||
|
v2_input,
|
||||||
|
v_parameterization_input,
|
||||||
|
train_dir_input,
|
||||||
|
image_folder_input,
|
||||||
|
output_dir_input,
|
||||||
|
logging_dir_input,
|
||||||
|
max_resolution_input,
|
||||||
|
learning_rate_input,
|
||||||
|
lr_scheduler_input,
|
||||||
|
lr_warmup_input,
|
||||||
|
dataset_repeats_input,
|
||||||
|
train_batch_size_input,
|
||||||
|
epoch_input,
|
||||||
|
save_every_n_epochs_input,
|
||||||
|
mixed_precision_input,
|
||||||
|
save_precision_input,
|
||||||
|
seed_input,
|
||||||
|
num_cpu_threads_per_process_input,
|
||||||
|
train_text_encoder_input,
|
||||||
|
create_buckets,
|
||||||
|
create_caption,
|
||||||
|
save_model_as_dropdown,
|
||||||
|
caption_extention_input,
|
||||||
|
],
|
||||||
|
outputs=[config_file_name],
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
# torch.cuda.set_per_process_memory_fraction(0.48)
|
# torch.cuda.set_per_process_memory_fraction(0.48)
|
||||||
parser = argparse.ArgumentParser()
|
parser = argparse.ArgumentParser()
|
||||||
parser.add_argument("--username", type=str, default='', help="Username for authentication")
|
parser.add_argument(
|
||||||
parser.add_argument("--password", type=str, default='', help="Password for authentication")
|
'--username', type=str, default='', help='Username for authentication'
|
||||||
|
)
|
||||||
args = parser.parse_args()
|
parser.add_argument(
|
||||||
|
'--password', type=str, default='', help='Password for authentication'
|
||||||
UI(username=args.username, password=args.password)
|
)
|
||||||
|
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
UI(username=args.username, password=args.password)
|
||||||
|
58
kohya_gui.py
Normal file
58
kohya_gui.py
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
import gradio as gr
|
||||||
|
import os
|
||||||
|
import argparse
|
||||||
|
from dreambooth_gui import dreambooth_tab
|
||||||
|
from finetune_gui import finetune_tab
|
||||||
|
from library.utilities import utilities_tab
|
||||||
|
|
||||||
|
|
||||||
|
def UI(username, password):
|
||||||
|
|
||||||
|
css = ''
|
||||||
|
|
||||||
|
if os.path.exists('./style.css'):
|
||||||
|
with open(os.path.join('./style.css'), 'r', encoding='utf8') as file:
|
||||||
|
print('Load CSS...')
|
||||||
|
css += file.read() + '\n'
|
||||||
|
|
||||||
|
interface = gr.Blocks(css=css)
|
||||||
|
|
||||||
|
with interface:
|
||||||
|
with gr.Tab('Dreambooth'):
|
||||||
|
(
|
||||||
|
train_data_dir_input,
|
||||||
|
reg_data_dir_input,
|
||||||
|
output_dir_input,
|
||||||
|
logging_dir_input,
|
||||||
|
) = dreambooth_tab()
|
||||||
|
with gr.Tab('Finetune'):
|
||||||
|
finetune_tab()
|
||||||
|
with gr.Tab('Utilities'):
|
||||||
|
utilities_tab(
|
||||||
|
train_data_dir_input=train_data_dir_input,
|
||||||
|
reg_data_dir_input=reg_data_dir_input,
|
||||||
|
output_dir_input=output_dir_input,
|
||||||
|
logging_dir_input=logging_dir_input,
|
||||||
|
enable_copy_info_button=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Show the interface
|
||||||
|
if not username == '':
|
||||||
|
interface.launch(auth=(username, password))
|
||||||
|
else:
|
||||||
|
interface.launch()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
# torch.cuda.set_per_process_memory_fraction(0.48)
|
||||||
|
parser = argparse.ArgumentParser()
|
||||||
|
parser.add_argument(
|
||||||
|
'--username', type=str, default='', help='Username for authentication'
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'--password', type=str, default='', help='Password for authentication'
|
||||||
|
)
|
||||||
|
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
UI(username=args.username, password=args.password)
|
@ -72,10 +72,13 @@ def get_saveasfile_path(file_path='', defaultextension='.json'):
|
|||||||
def add_pre_postfix(
|
def add_pre_postfix(
|
||||||
folder='', prefix='', postfix='', caption_file_ext='.caption'
|
folder='', prefix='', postfix='', caption_file_ext='.caption'
|
||||||
):
|
):
|
||||||
|
if prefix == '' and postfix == '':
|
||||||
|
return
|
||||||
|
|
||||||
# set caption extention to default in case it was not provided
|
# set caption extention to default in case it was not provided
|
||||||
if caption_file_ext == '':
|
if caption_file_ext == '':
|
||||||
caption_file_ext = '.caption'
|
caption_file_ext = '.caption'
|
||||||
|
|
||||||
files = [f for f in os.listdir(folder) if f.endswith(caption_file_ext)]
|
files = [f for f in os.listdir(folder) if f.endswith(caption_file_ext)]
|
||||||
if not prefix == '':
|
if not prefix == '':
|
||||||
prefix = f'{prefix} '
|
prefix = f'{prefix} '
|
||||||
|
@ -51,7 +51,12 @@ def dataset_balancing(concept_repeats, folder, insecure):
|
|||||||
if match:
|
if match:
|
||||||
# Multiply the repeats value by the number inside the braces
|
# Multiply the repeats value by the number inside the braces
|
||||||
if not images == 0:
|
if not images == 0:
|
||||||
repeats = max(1, round(concept_repeats / images * float(match.group(1))))
|
repeats = max(
|
||||||
|
1,
|
||||||
|
round(
|
||||||
|
concept_repeats / images * float(match.group(1))
|
||||||
|
),
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
repeats = 0
|
repeats = 0
|
||||||
subdir = subdir[match.end() :]
|
subdir = subdir[match.end() :]
|
||||||
@ -95,7 +100,7 @@ def warning(insecure):
|
|||||||
|
|
||||||
|
|
||||||
def gradio_dataset_balancing_tab():
|
def gradio_dataset_balancing_tab():
|
||||||
with gr.Tab('Dataset balancing'):
|
with gr.Tab('Dreambooth Dataset balancing'):
|
||||||
gr.Markdown(
|
gr.Markdown(
|
||||||
'This utility will ensure that each concept folder in the dataset folder is used equally during the training process of the dreambooth machine learning model, regardless of the number of images in each folder. It will do this by renaming the concept folders to indicate the number of times they should be repeated during training.'
|
'This utility will ensure that each concept folder in the dataset folder is used equally during the training process of the dreambooth machine learning model, regardless of the number of images in each folder. It will do this by renaming the concept folders to indicate the number of times they should be repeated during training.'
|
||||||
)
|
)
|
||||||
|
@ -68,31 +68,31 @@ def dreambooth_folder_preparation(
|
|||||||
print(f'Copy {util_training_images_dir_input} to {training_dir}...')
|
print(f'Copy {util_training_images_dir_input} to {training_dir}...')
|
||||||
shutil.copytree(util_training_images_dir_input, training_dir)
|
shutil.copytree(util_training_images_dir_input, training_dir)
|
||||||
|
|
||||||
# Create the regularization_dir path
|
if not util_regularization_images_dir_input == '':
|
||||||
if (
|
# Create the regularization_dir path
|
||||||
util_class_prompt_input == ''
|
if not util_regularization_images_repeat_input > 0:
|
||||||
or not util_regularization_images_repeat_input > 0
|
print('Repeats is missing... not copying regularisation images...')
|
||||||
):
|
else:
|
||||||
print(
|
regularization_dir = os.path.join(
|
||||||
'Regularization images directory or repeats is missing... not copying regularisation images...'
|
util_training_dir_output,
|
||||||
)
|
f'reg/{int(util_regularization_images_repeat_input)}_{util_class_prompt_input}',
|
||||||
|
)
|
||||||
|
|
||||||
|
# Remove folders if they exist
|
||||||
|
if os.path.exists(regularization_dir):
|
||||||
|
print(f'Removing existing directory {regularization_dir}...')
|
||||||
|
shutil.rmtree(regularization_dir)
|
||||||
|
|
||||||
|
# Copy the regularisation images to their respective directories
|
||||||
|
print(
|
||||||
|
f'Copy {util_regularization_images_dir_input} to {regularization_dir}...'
|
||||||
|
)
|
||||||
|
shutil.copytree(
|
||||||
|
util_regularization_images_dir_input, regularization_dir
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
regularization_dir = os.path.join(
|
|
||||||
util_training_dir_output,
|
|
||||||
f'reg/{int(util_regularization_images_repeat_input)}_{util_class_prompt_input}',
|
|
||||||
)
|
|
||||||
|
|
||||||
# Remove folders if they exist
|
|
||||||
if os.path.exists(regularization_dir):
|
|
||||||
print(f'Removing existing directory {regularization_dir}...')
|
|
||||||
shutil.rmtree(regularization_dir)
|
|
||||||
|
|
||||||
# Copy the regularisation images to their respective directories
|
|
||||||
print(
|
print(
|
||||||
f'Copy {util_regularization_images_dir_input} to {regularization_dir}...'
|
'Regularization images directory is missing... not copying regularisation images...'
|
||||||
)
|
|
||||||
shutil.copytree(
|
|
||||||
util_regularization_images_dir_input, regularization_dir
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# create log and model folder
|
# create log and model folder
|
||||||
@ -110,10 +110,11 @@ def dreambooth_folder_preparation(
|
|||||||
|
|
||||||
|
|
||||||
def gradio_dreambooth_folder_creation_tab(
|
def gradio_dreambooth_folder_creation_tab(
|
||||||
train_data_dir_input,
|
train_data_dir_input=gr.Textbox(),
|
||||||
reg_data_dir_input,
|
reg_data_dir_input=gr.Textbox(),
|
||||||
output_dir_input,
|
output_dir_input=gr.Textbox(),
|
||||||
logging_dir_input,
|
logging_dir_input=gr.Textbox(),
|
||||||
|
enable_copy_info_button=bool(False),
|
||||||
):
|
):
|
||||||
with gr.Tab('Dreambooth folder preparation'):
|
with gr.Tab('Dreambooth folder preparation'):
|
||||||
gr.Markdown(
|
gr.Markdown(
|
||||||
@ -191,16 +192,17 @@ def gradio_dreambooth_folder_creation_tab(
|
|||||||
util_training_dir_output,
|
util_training_dir_output,
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
button_copy_info_to_Directories_tab = gr.Button(
|
if enable_copy_info_button:
|
||||||
'Copy info to Directories Tab'
|
button_copy_info_to_Directories_tab = gr.Button(
|
||||||
)
|
'Copy info to Directories Tab'
|
||||||
button_copy_info_to_Directories_tab.click(
|
)
|
||||||
copy_info_to_Directories_tab,
|
button_copy_info_to_Directories_tab.click(
|
||||||
inputs=[util_training_dir_output],
|
copy_info_to_Directories_tab,
|
||||||
outputs=[
|
inputs=[util_training_dir_output],
|
||||||
train_data_dir_input,
|
outputs=[
|
||||||
reg_data_dir_input,
|
train_data_dir_input,
|
||||||
output_dir_input,
|
reg_data_dir_input,
|
||||||
logging_dir_input,
|
output_dir_input,
|
||||||
],
|
logging_dir_input,
|
||||||
)
|
],
|
||||||
|
)
|
||||||
|
84
library/utilities.py
Normal file
84
library/utilities.py
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
# v1: initial release
|
||||||
|
# v2: add open and save folder icons
|
||||||
|
# v3: Add new Utilities tab for Dreambooth folder preparation
|
||||||
|
# v3.1: Adding captionning of images to utilities
|
||||||
|
|
||||||
|
import gradio as gr
|
||||||
|
import os
|
||||||
|
import argparse
|
||||||
|
from library.dreambooth_folder_creation_gui import (
|
||||||
|
gradio_dreambooth_folder_creation_tab,
|
||||||
|
)
|
||||||
|
from library.basic_caption_gui import gradio_basic_caption_gui_tab
|
||||||
|
from library.convert_model_gui import gradio_convert_model_tab
|
||||||
|
from library.blip_caption_gui import gradio_blip_caption_gui_tab
|
||||||
|
from library.wd14_caption_gui import gradio_wd14_caption_gui_tab
|
||||||
|
from library.dataset_balancing_gui import gradio_dataset_balancing_tab
|
||||||
|
|
||||||
|
|
||||||
|
def utilities_tab(
|
||||||
|
train_data_dir_input=gr.Textbox(),
|
||||||
|
reg_data_dir_input=gr.Textbox(),
|
||||||
|
output_dir_input=gr.Textbox(),
|
||||||
|
logging_dir_input=gr.Textbox(),
|
||||||
|
enable_copy_info_button=bool(False),
|
||||||
|
enable_dreambooth_tab=True,
|
||||||
|
):
|
||||||
|
with gr.Tab('Captioning'):
|
||||||
|
gradio_basic_caption_gui_tab()
|
||||||
|
gradio_blip_caption_gui_tab()
|
||||||
|
gradio_wd14_caption_gui_tab()
|
||||||
|
if enable_dreambooth_tab:
|
||||||
|
with gr.Tab('Dreambooth'):
|
||||||
|
gr.Markdown('This section provide Dreambooth specific tools.')
|
||||||
|
gradio_dreambooth_folder_creation_tab(
|
||||||
|
train_data_dir_input=train_data_dir_input,
|
||||||
|
reg_data_dir_input=reg_data_dir_input,
|
||||||
|
output_dir_input=output_dir_input,
|
||||||
|
logging_dir_input=logging_dir_input,
|
||||||
|
enable_copy_info_button=enable_copy_info_button,
|
||||||
|
)
|
||||||
|
gradio_dataset_balancing_tab()
|
||||||
|
gradio_convert_model_tab()
|
||||||
|
|
||||||
|
return (
|
||||||
|
train_data_dir_input,
|
||||||
|
reg_data_dir_input,
|
||||||
|
output_dir_input,
|
||||||
|
logging_dir_input,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def UI(username, password):
|
||||||
|
css = ''
|
||||||
|
|
||||||
|
if os.path.exists('./style.css'):
|
||||||
|
with open(os.path.join('./style.css'), 'r', encoding='utf8') as file:
|
||||||
|
print('Load CSS...')
|
||||||
|
css += file.read() + '\n'
|
||||||
|
|
||||||
|
interface = gr.Blocks(css=css)
|
||||||
|
|
||||||
|
with interface:
|
||||||
|
utilities_tab()
|
||||||
|
|
||||||
|
# Show the interface
|
||||||
|
if not username == '':
|
||||||
|
interface.launch(auth=(username, password))
|
||||||
|
else:
|
||||||
|
interface.launch()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
# torch.cuda.set_per_process_memory_fraction(0.48)
|
||||||
|
parser = argparse.ArgumentParser()
|
||||||
|
parser.add_argument(
|
||||||
|
'--username', type=str, default='', help='Username for authentication'
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'--password', type=str, default='', help='Password for authentication'
|
||||||
|
)
|
||||||
|
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
UI(username=args.username, password=args.password)
|
1
utilities.cmd
Normal file
1
utilities.cmd
Normal file
@ -0,0 +1 @@
|
|||||||
|
.\venv\Scripts\python.exe library\utilities.py
|
Loading…
Reference in New Issue
Block a user