# Copyright 2024 Bytedance Ltd. and/or its affiliates # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # http://www.apache.org/licenses/LICENSE-2.0 # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. """ Multi-turn SFT dataset that supports training on conversation data with multiple turns """ from typing import List, Union import pandas as pd import torch from torch.utils.data import Dataset from transformers import PreTrainedTokenizer from verl.utils import hf_tokenizer from verl.utils.fs import copy_local_path_from_hdfs class MultiTurnSFTDataset(Dataset): """ Dataset for multi-turn conversations where each assistant response should be trained """ def __init__(self, parquet_files: Union[str, List[str]], tokenizer, config=None): # Set defaults and extract parameters from config if provided config = config or {} self.truncation = config.get("truncation", "error") self.max_length = config.get("max_length", 1024) # Get messages_key from the new multiturn config structure multiturn_config = config.get("multiturn", {}) self.messages_key = multiturn_config.get("messages_key", "messages") assert self.truncation in ["error", "left", "right"] if not isinstance(parquet_files, List): parquet_files = [parquet_files] self.parquet_files = parquet_files if isinstance(tokenizer, str): tokenizer = hf_tokenizer(tokenizer) self.tokenizer: PreTrainedTokenizer = tokenizer self._download() self._read_files_and_process() def _download(self): for i, parquet_file in enumerate(self.parquet_files): self.parquet_files[i] = copy_local_path_from_hdfs(parquet_file, verbose=True) def _read_files_and_process(self): def series_to_item(ls): import numpy import pandas while isinstance(ls, (pandas.core.series.Series, numpy.ndarray)) and len(ls) == 1: ls = ls[0] return ls dataframes = [] for parquet_file in self.parquet_files: dataframe = pd.read_parquet(parquet_file) dataframes.append(dataframe) self.dataframe = pd.concat(dataframes) # Extract messages list from dataframe self.messages = self.dataframe[self.messages_key].apply(series_to_item).tolist() def __len__(self): return len(self.messages) def __getitem__(self, item): tokenizer = self.tokenizer messages = self.messages[item] # First, get the full conversation tokens full_tokens = tokenizer.apply_chat_template(messages, tokenize=True, return_tensors="pt", add_generation_prompt=False) input_ids = full_tokens[0] # The output is already a tensor attention_mask = torch.ones_like(input_ids) # Create loss mask by identifying assistant responses loss_mask = torch.zeros_like(input_ids, dtype=torch.long) # Process each message to find assistant responses for i, msg in enumerate(messages): # Get tokens for messages up to this point to find the start position prefix_messages = messages[: i + 1] prefix_tokens = tokenizer.apply_chat_template(prefix_messages, tokenize=True, return_tensors="pt", add_generation_prompt=False) # Get tokens for messages up to previous point prev_tokens = tokenizer.apply_chat_template(messages[:i], tokenize=True, return_tensors="pt", add_generation_prompt=False) if i > 0 else None # Calculate start and end positions start_pos = prev_tokens[0].shape[0] if prev_tokens is not None else 0 end_pos = prefix_tokens[0].shape[0] # If this is an assistant message, set loss mask if msg["role"] == "assistant": loss_mask[start_pos:end_pos] = 1 # Handle sequence length sequence_length = input_ids.shape[0] if sequence_length < self.max_length: # Pad sequences pad_token_id = self.tokenizer.pad_token_id if self.tokenizer.pad_token_id is not None else 0 padded_input_ids = torch.ones(size=(self.max_length - sequence_length,), dtype=input_ids.dtype) * pad_token_id padded_attention_mask = torch.zeros(size=(self.max_length - sequence_length,), dtype=attention_mask.dtype) padded_loss_mask = torch.zeros(size=(self.max_length - sequence_length,), dtype=loss_mask.dtype) input_ids = torch.cat((input_ids, padded_input_ids)) attention_mask = torch.cat((attention_mask, padded_attention_mask)) loss_mask = torch.cat((loss_mask, padded_loss_mask)) elif sequence_length > self.max_length: if self.truncation == "left": input_ids = input_ids[-self.max_length :] attention_mask = attention_mask[-self.max_length :] loss_mask = loss_mask[-self.max_length :] elif self.truncation == "right": input_ids = input_ids[: self.max_length] attention_mask = attention_mask[: self.max_length] loss_mask = loss_mask[: self.max_length] elif self.truncation == "error": raise ValueError(f"{sequence_length=} is larger than {self.max_length=}") else: raise ValueError(f"Unknown truncation method {self.truncation}") # Create position IDs position_ids = torch.arange(len(input_ids), dtype=torch.long) # Zero out position IDs for padding position_ids = position_ids * attention_mask return { "input_ids": input_ids, "attention_mask": attention_mask, "position_ids": position_ids, "loss_mask": loss_mask, }