libkazz's picture
Hide script
c2fd441 verified
|
raw
history blame
6.24 kB
---
base_model: llm-jp/llm-jp-3-13b
tags:
- text-generation-inference
- transformers
- unsloth
- llama
- trl
license: apache-2.0
language:
- en
library_name: peft
---
# Uploaded model
- **Developed by:** libkazz
- **License:** apache-2.0
- **Finetuned from model :** llm-jp/llm-jp-3-13b
This llama model was trained 2x faster with [Unsloth](https://github.com/unslothai/unsloth) and Huggingface's TRL library.
[<img src="https://raw.githubusercontent.com/unslothai/unsloth/main/images/unsloth%20made%20with%20love.png" width="200"/>](https://github.com/unslothai/unsloth)
### Training Details
指示チューニングデータとして下記のものを利用しました。
* ichikara-instruction-003-001-1.json
* ichikara-instruction-003-002-1.json
* ichikara-instruction-003-003-1.json
以下について, 指示に対して複数の回答を持つデータは回答を1件のみ抽出して指示データをして利用しました。
* ichikara-instruction-003-001-5.1.json
* ichikara-instruction-003-001-5.2.json
* ichikara-instruction-003-001-2.1.json
* ichikara-instruction-003-001-2.2.json
回答を1件のみ抽出するために gpt-4o-mini で下記 script を利用しました。
<details>
<summary>script</summary>
```python
# gpt-4o-mini を使って最も適切そうな回答を選定
def select_best_response(file_path, output_dir):
with open(file_path, "r", encoding="utf-8") as file:
data = json.load(file)
grouped_data = {}
# データをデータ番号ごとに分類
for item in data:
data_number = item["ID"].split("-")[4]
grouped_data.setdefault(data_number, []).append(item)
filtered_data = []
for data_number, responses in grouped_data.items():
print(f"\n#### {data_number} ####")
# 指示と回答リストを生成
instructions = [r["text"] for r in responses]
outputs = [r["output"] for r in responses]
# GPT-4o-miniに最適な回答を選定させる
prompt = (
"以下の指示に対して最も適切な回答を選定してください。\n\n"
f"### 指示 ###\n```\n{instructions[0]}\n```\n\n"
"### 候補 ###\n" + "\n\n".join([f"{i+1}: \n```\n{output}\n```" for i, output in enumerate(outputs)]) +
"\n\n### 回答 ###\n番号(1 or 2)のみを回答してください。1つだけ選択してください。\n"
)
try:
response = openai.chat.completions.create(
model="gpt-4o-mini",
messages=[
{"role": "system", "content": "あなたはIQ140の論理的思考力の高い文章評価の専門家です。"},
{"role": "user", "content": prompt}
]
)
content = response.choices[0].message.content
print(content)
best_choice = int(content.strip()) - 1
except ValueError as e:
print(f"ValueError: 無効な応答がありました。データ番号: {data_number}")
print(f"応答内容: {response.choices[0].message['content']}")
best_choice = 1
except Exception as e:
print(f"予期しないエラーが発生しました。データ番号: {data_number}")
print(e)
best_choice = 1
filtered_data.append(responses[best_choice])
output_file_name = os.path.basename(file_path)
output_file_path = os.path.join(output_dir, output_file_name)
with open(output_file_path, "w", encoding="utf-8") as output_file:
json.dump(filtered_data, output_file, ensure_ascii=False, indent=2)
```
</details>
### ライセンス
* ichikara-instruction データセットのライセンスは cc-by-nc-sa になっております。
### SFTの概要
* 4bit量子化
* LoRAによるSFT
* learning_rate = 2e-4
* num_train_epochs = 2
## Bias, Risks, and Limitations
RLHF,DPOを実施していないため不適切な表現が出力される可能性があります。
# elyza-tasks-100-TV_0.jsonl の出力方法
elyza-tasks-100-TV_0.jsonl に記載されている指示に対する返答のサンプル出力コードは次のようになります。
```python
import torch
from transformers import (
AutoTokenizer,
AutoModelForCausalLM,
BitsAndBytesConfig,
)
from peft import LoraConfig, PeftModel
from datasets import load_dataset
BASE_MODEL = "llm-jp/llm-jp-3-13b"
PEFT_MODEL = "libkazz/llm-jp-3-13b-it"
tokenizer = AutoTokenizer.from_pretrained(PEFT_MODEL)
bnb_config = BitsAndBytesConfig(
load_in_4bit=True,
bnb_4bit_compute_dtype=torch.float16,
bnb_4bit_quant_type="nf4",
bnb_4bit_use_double_quant=False,
)
base_model = AutoModelForCausalLM.from_pretrained(
BASE_MODEL,
device_map="auto",
quantization_config=bnb_config,
torch_dtype="auto",
trust_remote_code=True,
)
model = PeftModel.from_pretrained(base_model, PEFT_MODEL)
# elyza-tasks-100-TV_0.jsonl データの読み込み
from datasets import load_dataset
dataset = load_dataset("json", data_files="./elyza-tasks-100-TV_0.jsonl", split="train")
results = []
for num in tqdm(range(100)):
instruction = dataset["input"][num]
prompt = f"次の指示に忠実に回答を作成しなさい。\n\n### 指示:\n{instruction}\n\n### 回答:\n"
model_input = tokenizer(prompt, return_tensors="pt").to(model.device)
input_ids = model_input["input_ids"]
with torch.no_grad():
outputs = model.generate(
input_ids,
max_new_tokens=300,
attention_mask = model_input.attention_mask,
pad_token_id=tokenizer.pad_token_id,
eos_token_id=tokenizer.eos_token_id,
do_sample=False,
repetition_penalty=1.02,
)[0]
output = tokenizer.decode(outputs[input_ids.size(1):], skip_special_tokens=True)
results.append({"task_id": num, "input": instruction, "output": output})
# ファイルに保存する
import json
with open("output.jsonl", "wt", encoding='utf-8') as f:
for result in results:
json.dump(result, f, ensure_ascii=False)
f.write('\n')
```