-
Notifications
You must be signed in to change notification settings - Fork 49
/
Copy pathtrain.py
512 lines (444 loc) · 19.4 KB
/
train.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
import os
os.environ["HF_HUB_DISABLE_TELEMETRY"] = "1"
os.environ["HF_HUB_ENABLE_HF_TRANSFER"] = "1"
import sys
sys.path.append("ai-toolkit")
sys.path.append("LLaVA")
from submodule_patches import patch_submodules
patch_submodules()
import shutil
import subprocess
import sys
import time
from typing import Optional, OrderedDict
from zipfile import ZipFile, is_zipfile
import torch
from cog import BaseModel, Input, Path, Secret # pyright: ignore
from extensions_built_in.sd_trainer.SDTrainer import SDTrainer
from huggingface_hub import HfApi
from jobs import BaseJob
from toolkit.config import get_config
from caption import Captioner
from wandb_client import WeightsAndBiasesClient, logout_wandb
from layer_match import match_layers_to_optimize, available_layers_to_optimize
JOB_NAME = "flux_train_replicate"
WEIGHTS_PATH = Path("./FLUX.1-dev")
INPUT_DIR = Path("input_images")
OUTPUT_DIR = Path("output")
JOB_DIR = OUTPUT_DIR / JOB_NAME
print(f"Environment language: {os.environ.get('LANG', 'Not set')}")
os.environ["LANG"] = "en_US.UTF-8"
print(f"Updated environment language: {os.environ.get('LANG', 'Not set')}")
class CustomSDTrainer(SDTrainer):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.seen_samples = set()
self.wandb: WeightsAndBiasesClient | None = None
def hook_train_loop(self, batch):
loss_dict = super().hook_train_loop(batch)
if self.wandb:
self.wandb.log_loss(loss_dict, self.step_num)
return loss_dict
def sample(self, step=None, is_first=False):
super().sample(step=step, is_first=is_first)
output_dir = JOB_DIR / "samples"
all_samples = set([p.name for p in output_dir.glob("*.jpg")])
new_samples = all_samples - self.seen_samples
if self.wandb:
image_paths = [output_dir / p for p in sorted(new_samples)]
self.wandb.log_samples(image_paths, step)
self.seen_samples = all_samples
def post_save_hook(self, save_path):
super().post_save_hook(save_path)
# final lora path
lora_path = JOB_DIR / f"{JOB_NAME}.safetensors"
if not lora_path.exists():
# intermediate saved weights
lora_path = sorted(JOB_DIR.glob("*.safetensors"))[-1]
if self.wandb:
print(f"Saving weights to W&B: {lora_path.name}")
self.wandb.save_weights(lora_path)
class CustomJob(BaseJob):
def __init__(
self, config: OrderedDict, wandb_client: WeightsAndBiasesClient | None
):
super().__init__(config)
self.device = self.get_conf("device", "cpu")
self.process_dict = {"custom_sd_trainer": CustomSDTrainer}
self.load_processes(self.process_dict)
for process in self.process:
process.wandb = wandb_client
def run(self):
super().run()
# Keeping this for backwards compatibility
print(
f"Running {len(self.process)} process{'' if len(self.process) == 1 else 'es'}"
)
for process in self.process:
process.run()
class TrainingOutput(BaseModel):
weights: Path
def train(
input_images: Path = Input(
description="A zip file containing the images that will be used for training. We recommend a minimum of 10 images. If you include captions, include them as one .txt file per image, e.g. my-photo.jpg should have a caption file named my-photo.txt. If you don't include captions, you can use autocaptioning (enabled by default).",
default=None,
),
trigger_word: str = Input(
description="The trigger word refers to the object, style or concept you are training on. Pick a string that isn’t a real word, like TOK or something related to what’s being trained, like CYBRPNK. The trigger word you specify here will be associated with all images during training. Then when you use your LoRA, you can include the trigger word in prompts to help activate the LoRA.",
default="TOK",
),
autocaption: bool = Input(
description="Automatically caption images using Llava v1.5 13B", default=True
),
autocaption_prefix: str = Input(
description="Optional: Text you want to appear at the beginning of all your generated captions; for example, ‘a photo of TOK, ’. You can include your trigger word in the prefix. Prefixes help set the right context for your captions, and the captioner will use this prefix as context.",
default=None,
),
autocaption_suffix: str = Input(
description="Optional: Text you want to appear at the end of all your generated captions; for example, ‘ in the style of TOK’. You can include your trigger word in suffixes. Suffixes help set the right concept for your captions, and the captioner will use this suffix as context.",
default=None,
),
steps: int = Input(
description="Number of training steps. Recommended range 500-4000",
ge=3,
le=6000,
default=1000,
),
learning_rate: float = Input(
description="Learning rate, if you’re new to training you probably don’t need to change this.",
default=4e-4,
),
batch_size: int = Input(
description="Batch size, you can leave this as 1", default=1
),
resolution: str = Input(
description="Image resolutions for training", default="512,768,1024"
),
lora_rank: int = Input(
description="Higher ranks take longer to train but can capture more complex features. Caption quality is more important for higher ranks.",
default=16,
ge=1,
le=128,
),
caption_dropout_rate: float = Input(
description="Advanced setting. Determines how often a caption is ignored. 0.05 means for 5% of all steps an image will be used without its caption. 0 means always use captions, while 1 means never use them. Dropping captions helps capture more details of an image, and can prevent over-fitting words with specific image elements. Try higher values when training a style.",
default=0.05,
ge=0,
le=1,
),
optimizer: str = Input(
description="Optimizer to use for training. Supports: prodigy, adam8bit, adamw8bit, lion8bit, adam, adamw, lion, adagrad, adafactor.",
default="adamw8bit",
),
cache_latents_to_disk: bool = Input(
description="Use this if you have lots of input images and you hit out of memory errors",
default=False,
),
layers_to_optimize_regex: str = Input(
description="Regular expression to match specific layers to optimize. Optimizing fewer layers results in shorter training times, but can also result in a weaker LoRA. For example, To target layers 7, 12, 16, 20 which seems to create good likeness with faster training (as discovered by lux in the Ostris discord, inspired by The Last Ben), use `transformer.single_transformer_blocks.(7|12|16|20).proj_out`.",
default=None,
),
hf_repo_id: str = Input(
description="Hugging Face repository ID, if you'd like to upload the trained LoRA to Hugging Face. For example, lucataco/flux-dev-lora. If the given repo does not exist, a new public repo will be created.",
default=None,
),
hf_token: Secret = Input(
description="Hugging Face token, if you'd like to upload the trained LoRA to Hugging Face.",
default=None,
),
wandb_api_key: Secret = Input(
description="Weights and Biases API key, if you'd like to log training progress to W&B.",
default=None,
),
wandb_project: str = Input(
description="Weights and Biases project name. Only applicable if wandb_api_key is set.",
default=JOB_NAME,
),
wandb_run: str = Input(
description="Weights and Biases run name. Only applicable if wandb_api_key is set.",
default=None,
),
wandb_entity: str = Input(
description="Weights and Biases entity name. Only applicable if wandb_api_key is set.",
default=None,
),
wandb_sample_interval: int = Input(
description="Step interval for sampling output images that are logged to W&B. Only applicable if wandb_api_key is set.",
default=100,
ge=1,
),
wandb_sample_prompts: str = Input(
description="Newline-separated list of prompts to use when logging samples to W&B. Only applicable if wandb_api_key is set.",
default=None,
),
wandb_save_interval: int = Input(
description="Step interval for saving intermediate LoRA weights to W&B. Only applicable if wandb_api_key is set.",
default=100,
ge=1,
),
skip_training_and_use_pretrained_hf_lora_url: str = Input(
description="If you’d like to skip LoRA training altogether and instead create a Replicate model from a pre-trained LoRA that’s on HuggingFace, use this field with a HuggingFace download URL. For example, https://huggingface.co/fofr/flux-80s-cyberpunk/resolve/main/lora.safetensors.",
default=None,
),
) -> TrainingOutput:
clean_up()
output_path = "/tmp/trained_model.tar"
if skip_training_and_use_pretrained_hf_lora_url is not None:
download_huggingface_lora(
skip_training_and_use_pretrained_hf_lora_url, output_path
)
return TrainingOutput(weights=Path(output_path))
if not input_images:
raise ValueError("input_images must be provided")
layers_to_optimize = None
if layers_to_optimize_regex:
layers_to_optimize = match_layers_to_optimize(layers_to_optimize_regex)
if not layers_to_optimize:
raise ValueError(
f"The regex '{layers_to_optimize_regex}' didn't match any layers. These layers can be optimized:\n"
+ "\n".join(available_layers_to_optimize)
)
sample_prompts = []
if wandb_sample_prompts:
sample_prompts = [p.strip() for p in wandb_sample_prompts.split("\n")]
train_config = OrderedDict(
{
"job": "custom_job",
"config": {
"name": JOB_NAME,
"process": [
{
"type": "custom_sd_trainer",
"training_folder": str(OUTPUT_DIR),
"device": "cuda:0",
"trigger_word": trigger_word,
"network": {
"type": "lora",
"linear": lora_rank,
"linear_alpha": lora_rank,
},
"save": {
"dtype": "float16",
"save_every": (
wandb_save_interval if wandb_api_key else steps + 1
),
"max_step_saves_to_keep": 1,
},
"datasets": [
{
"folder_path": str(INPUT_DIR),
"caption_ext": "txt",
"caption_dropout_rate": caption_dropout_rate,
"shuffle_tokens": False,
# TODO: Do we need to cache to disk? It's faster not to.
"cache_latents_to_disk": cache_latents_to_disk,
"cache_latents": True,
"resolution": [
int(res) for res in resolution.split(",")
],
}
],
"train": {
"batch_size": batch_size,
"steps": steps,
"gradient_accumulation_steps": 1,
"train_unet": True,
"train_text_encoder": False,
"content_or_style": "balanced",
"gradient_checkpointing": True,
"noise_scheduler": "flowmatch",
"optimizer": optimizer,
"lr": learning_rate,
"ema_config": {"use_ema": True, "ema_decay": 0.99},
"dtype": "bf16",
},
"model": {
"name_or_path": str(WEIGHTS_PATH),
"is_flux": True,
"quantize": True,
},
"sample": {
"sampler": "flowmatch",
"sample_every": (
wandb_sample_interval
if wandb_api_key and sample_prompts
else steps + 1
),
"width": 1024,
"height": 1024,
"prompts": sample_prompts,
"neg": "",
"seed": 42,
"walk_seed": True,
"guidance_scale": 3.5,
"sample_steps": 28,
},
}
],
},
"meta": {"name": "[name]", "version": "1.0"},
}
)
if layers_to_optimize:
train_config["config"]["process"][0]["network"]["network_kwargs"] = {
"only_if_contains": layers_to_optimize
}
wandb_client = None
if wandb_api_key:
wandb_config = {
"trigger_word": trigger_word,
"autocaption": autocaption,
"autocaption_prefix": autocaption_prefix,
"autocaption_suffix": autocaption_suffix,
"steps": steps,
"learning_rate": learning_rate,
"batch_size": batch_size,
"resolution": resolution,
"lora_rank": lora_rank,
"caption_dropout_rate": caption_dropout_rate,
"optimizer": optimizer,
}
wandb_client = WeightsAndBiasesClient(
api_key=wandb_api_key.get_secret_value(),
config=wandb_config,
sample_prompts=sample_prompts,
project=wandb_project,
entity=wandb_entity,
name=wandb_run,
)
download_weights()
extract_zip(input_images, INPUT_DIR)
if not trigger_word:
del train_config["config"]["process"][0]["trigger_word"]
captioner = Captioner()
if autocaption and not captioner.all_images_are_captioned(INPUT_DIR):
captioner.load_models()
captioner.caption_images(INPUT_DIR, autocaption_prefix, autocaption_suffix)
del captioner
torch.cuda.empty_cache()
print("Starting train job")
job = CustomJob(get_config(train_config, name=None), wandb_client)
job.run()
if wandb_client:
wandb_client.finish()
job.cleanup()
lora_file = JOB_DIR / f"{JOB_NAME}.safetensors"
lora_file.rename(JOB_DIR / "lora.safetensors")
samples_dir = JOB_DIR / "samples"
if samples_dir.exists():
shutil.rmtree(samples_dir)
# Remove any intermediate lora paths
lora_paths = JOB_DIR.glob("*.safetensors")
for path in lora_paths:
if path.name != "lora.safetensors":
path.unlink()
# Optimizer is used to continue training, not needed in output
optimizer_file = JOB_DIR / "optimizer.pt"
if optimizer_file.exists():
optimizer_file.unlink()
# Copy generated captions to the output tar
# But do not upload publicly to HF
captions_dir = JOB_DIR / "captions"
captions_dir.mkdir(exist_ok=True)
for caption_file in INPUT_DIR.glob("*.txt"):
shutil.copy(caption_file, captions_dir)
os.system(f"tar -cvf {output_path} {JOB_DIR}")
if hf_token is not None and hf_repo_id is not None:
if captions_dir.exists():
shutil.rmtree(captions_dir)
try:
handle_hf_readme(hf_repo_id, trigger_word)
print(f"Uploading to Hugging Face: {hf_repo_id}")
api = HfApi()
repo_url = api.create_repo(
hf_repo_id,
private=False,
exist_ok=True,
token=hf_token.get_secret_value(),
)
print(f"HF Repo URL: {repo_url}")
api.upload_folder(
repo_id=hf_repo_id,
folder_path=JOB_DIR,
repo_type="model",
use_auth_token=hf_token.get_secret_value(),
)
except Exception as e:
print(f"Error uploading to Hugging Face: {str(e)}")
return TrainingOutput(weights=Path(output_path))
def handle_hf_readme(hf_repo_id: str, trigger_word: Optional[str]):
readme_path = JOB_DIR / "README.md"
license_path = Path("lora-license.md")
shutil.copy(license_path, readme_path)
content = readme_path.read_text()
content = content.replace("[hf_repo_id]", hf_repo_id)
repo_parts = hf_repo_id.split("/")
if len(repo_parts) > 1:
title = repo_parts[1].replace("-", " ").title()
content = content.replace("[title]", title)
else:
content = content.replace("[title]", hf_repo_id)
if trigger_word:
content = content.replace(
"[trigger_section]",
f"\n## Trigger words\nYou should use `{trigger_word}` to trigger the image generation.\n",
)
content = content.replace(
"[instance_prompt]", f"instance_prompt: {trigger_word}"
)
else:
content = content.replace("[trigger_section]", "")
content = content.replace("[instance_prompt]", "")
print(content)
readme_path.write_text(content)
def extract_zip(input_images: Path, input_dir: Path):
if not is_zipfile(input_images):
raise ValueError("input_images must be a zip file")
input_dir.mkdir(parents=True, exist_ok=True)
image_count = 0
with ZipFile(input_images, "r") as zip_ref:
for file_info in zip_ref.infolist():
if not file_info.filename.startswith(
"__MACOSX/"
) and not file_info.filename.startswith("._"):
zip_ref.extract(file_info, input_dir)
image_count += 1
print(f"Extracted {image_count} files from zip to {input_dir}")
def clean_up():
logout_wandb()
if INPUT_DIR.exists():
shutil.rmtree(INPUT_DIR)
if OUTPUT_DIR.exists():
shutil.rmtree(OUTPUT_DIR)
def download_huggingface_lora(hf_lora_url: str, output_path: str):
if (
not hf_lora_url.startswith("https://huggingface.co")
or ".safetensors" not in hf_lora_url
):
raise ValueError(
"Invalid URL. Use a HuggingFace download URL like https://huggingface.co/fofr/flux-80s-cyberpunk/resolve/main/lora.safetensors"
)
lora_path = OUTPUT_DIR / "flux_train_replicate" / "lora.safetensors"
print(f"Downloading {hf_lora_url} to {lora_path}")
subprocess.check_output(
[
"pget",
"-f",
hf_lora_url,
lora_path,
]
)
os.system(f"tar -cvf {output_path} {lora_path}")
def download_weights():
if not WEIGHTS_PATH.exists():
t1 = time.time()
subprocess.check_output(
[
"pget",
"-xf",
"https://weights.replicate.delivery/default/black-forest-labs/FLUX.1-dev/files.tar",
str(WEIGHTS_PATH.parent),
]
)
t2 = time.time()
print(f"Downloaded base weights in {t2 - t1} seconds")