| | import os |
| |
|
| | import torch |
| | from datasets import load_dataset |
| | from torch.utils.data import DataLoader |
| | from tqdm import tqdm |
| | from transformers import AutoModelForSeq2SeqLM, AutoTokenizer, default_data_collator, get_linear_schedule_with_warmup |
| |
|
| | from peft import AdaLoraConfig, PeftConfig, PeftModel, TaskType, get_peft_model |
| |
|
| |
|
| | os.environ["TOKENIZERS_PARALLELISM"] = "false" |
| |
|
| | device = "cuda" |
| | model_name_or_path = "facebook/bart-base" |
| | tokenizer_name_or_path = "facebook/bart-base" |
| |
|
| | checkpoint_name = "financial_sentiment_analysis_lora_v1.pt" |
| | text_column = "sentence" |
| | label_column = "text_label" |
| | max_length = 128 |
| | lr = 1e-3 |
| | num_epochs = 8 |
| | batch_size = 8 |
| |
|
| |
|
| | |
| | peft_config = AdaLoraConfig( |
| | init_r=12, |
| | target_r=8, |
| | beta1=0.85, |
| | beta2=0.85, |
| | tinit=200, |
| | tfinal=1000, |
| | deltaT=10, |
| | lora_alpha=32, |
| | lora_dropout=0.1, |
| | task_type=TaskType.SEQ_2_SEQ_LM, |
| | inference_mode=False, |
| | ) |
| |
|
| | model = AutoModelForSeq2SeqLM.from_pretrained(model_name_or_path) |
| | model = get_peft_model(model, peft_config) |
| | model.print_trainable_parameters() |
| |
|
| |
|
| | |
| | dataset = load_dataset("financial_phrasebank", "sentences_allagree") |
| | dataset = dataset["train"].train_test_split(test_size=0.1) |
| | dataset["validation"] = dataset["test"] |
| | del dataset["test"] |
| |
|
| | classes = dataset["train"].features["label"].names |
| | dataset = dataset.map( |
| | lambda x: {"text_label": [classes[label] for label in x["label"]]}, |
| | batched=True, |
| | num_proc=1, |
| | ) |
| |
|
| |
|
| | |
| | tokenizer = AutoTokenizer.from_pretrained(model_name_or_path) |
| |
|
| |
|
| | def preprocess_function(examples): |
| | inputs = examples[text_column] |
| | targets = examples[label_column] |
| | model_inputs = tokenizer(inputs, max_length=max_length, padding="max_length", truncation=True, return_tensors="pt") |
| | labels = tokenizer(targets, max_length=3, padding="max_length", truncation=True, return_tensors="pt") |
| | labels = labels["input_ids"] |
| | labels[labels == tokenizer.pad_token_id] = -100 |
| | model_inputs["labels"] = labels |
| | return model_inputs |
| |
|
| |
|
| | processed_datasets = dataset.map( |
| | preprocess_function, |
| | batched=True, |
| | num_proc=1, |
| | remove_columns=dataset["train"].column_names, |
| | load_from_cache_file=False, |
| | desc="Running tokenizer on dataset", |
| | ) |
| |
|
| | train_dataset = processed_datasets["train"] |
| | eval_dataset = processed_datasets["validation"] |
| |
|
| | train_dataloader = DataLoader( |
| | train_dataset, shuffle=True, collate_fn=default_data_collator, batch_size=batch_size, pin_memory=True |
| | ) |
| | eval_dataloader = DataLoader(eval_dataset, collate_fn=default_data_collator, batch_size=batch_size, pin_memory=True) |
| |
|
| |
|
| | |
| | optimizer = torch.optim.AdamW(model.parameters(), lr=lr) |
| | lr_scheduler = get_linear_schedule_with_warmup( |
| | optimizer=optimizer, |
| | num_warmup_steps=0, |
| | num_training_steps=(len(train_dataloader) * num_epochs), |
| | ) |
| | model.base_model.peft_config["default"].total_step = len(train_dataloader) * num_epochs |
| |
|
| |
|
| | |
| | model = model.to(device) |
| | global_step = 0 |
| | for epoch in range(num_epochs): |
| | model.train() |
| | total_loss = 0 |
| | for step, batch in enumerate(tqdm(train_dataloader)): |
| | batch = {k: v.to(device) for k, v in batch.items()} |
| | outputs = model(**batch) |
| | loss = outputs.loss |
| | total_loss += loss.detach().float() |
| | loss.backward() |
| | optimizer.step() |
| | lr_scheduler.step() |
| | |
| | |
| | model.base_model.update_and_allocate(global_step) |
| | optimizer.zero_grad() |
| | global_step += 1 |
| |
|
| | model.eval() |
| | eval_loss = 0 |
| | eval_preds = [] |
| | for step, batch in enumerate(tqdm(eval_dataloader)): |
| | batch = {k: v.to(device) for k, v in batch.items()} |
| | with torch.no_grad(): |
| | outputs = model(**batch) |
| | loss = outputs.loss |
| | eval_loss += loss.detach().float() |
| | eval_preds.extend( |
| | tokenizer.batch_decode(torch.argmax(outputs.logits, -1).detach().cpu().numpy(), skip_special_tokens=True) |
| | ) |
| |
|
| | eval_epoch_loss = eval_loss / len(train_dataloader) |
| | eval_ppl = torch.exp(eval_epoch_loss) |
| | train_epoch_loss = total_loss / len(eval_dataloader) |
| | train_ppl = torch.exp(train_epoch_loss) |
| | print(f"{epoch=}: {train_ppl=} {train_epoch_loss=} {eval_ppl=} {eval_epoch_loss=}") |
| |
|
| |
|
| | |
| | correct = 0 |
| | total = 0 |
| | for pred, true in zip(eval_preds, dataset["validation"]["text_label"]): |
| | if pred.strip() == true.strip(): |
| | correct += 1 |
| | total += 1 |
| | accuracy = correct / total * 100 |
| | print(f"{accuracy=} % on the evaluation dataset") |
| | print(f"{eval_preds[:10]=}") |
| | print(f"{dataset['validation']['text_label'][:10]=}") |
| |
|
| |
|
| | |
| | peft_model_id = f"{model_name_or_path}_{peft_config.peft_type}_{peft_config.task_type}" |
| | model.save_pretrained(peft_model_id) |
| |
|
| |
|
| | ckpt = f"{peft_model_id}/adapter_model.bin" |
| | |
| |
|
| |
|
| | peft_model_id = f"{model_name_or_path}_{peft_config.peft_type}_{peft_config.task_type}" |
| |
|
| | config = PeftConfig.from_pretrained(peft_model_id) |
| | model = AutoModelForSeq2SeqLM.from_pretrained(config.base_model_name_or_path) |
| | model = PeftModel.from_pretrained(model, peft_model_id) |
| |
|
| |
|
| | model.eval() |
| | i = 13 |
| | inputs = tokenizer(dataset["validation"][text_column][i], return_tensors="pt") |
| | print(dataset["validation"][text_column][i]) |
| | print(inputs) |
| |
|
| | with torch.no_grad(): |
| | outputs = model.generate(input_ids=inputs["input_ids"], max_new_tokens=10) |
| | print(outputs) |
| | print(tokenizer.batch_decode(outputs.detach().cpu().numpy(), skip_special_tokens=True)) |
| |
|