Spaces:
Running
Running
File size: 7,095 Bytes
650c5f6 |
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 |
# Discriminative Reranking for Neural Machine Translation
https://aclanthology.org/2021.acl-long.563/
This folder contains source code for training DrNMT, a discriminatively trained reranker for neural machine translation.
## Data preparation
1. Follow the instructions under `examples/translation` to build a base MT model. Prepare three files, one with source sentences, one with ground truth target sentences, and one with hypotheses generated from the base MT model. Each line in the file contains one sentence in raw text (i.e. no sentencepiece, etc.). Below is an example of the files with _N_ hypotheses for each source sentence.
```
# Example of the source sentence file: (The file should contain L lines.)
source_sentence_1
source_sentence_2
source_sentence_3
...
source_sentence_L
# Example of the target sentence file: (The file should contain L lines.)
target_sentence_1
target_sentence_2
target_sentence_3
...
target_sentence_L
# Example of the hypotheses file: (The file should contain L*N lines.)
source_sentence_1_hypo_1
source_sentence_1_hypo_2
...
source_sentence_1_hypo_N
source_sentence_2_hypo_1
...
source_sentence_2_hypo_N
...
source_sentence_L_hypo_1
...
source_sentence_L_hypo_N
```
2. Download the [XLMR model](https://github.com/fairinternal/fairseq-py/tree/main/examples/xlmr#pre-trained-models).
```
wget https://dl.fbaipublicfiles.com/fairseq/models/xlmr.base.tar.gz
tar zxvf xlmr.base.tar.gz
# The folder should contain dict.txt, model.pt and sentencepiece.bpe.model.
```
3. Prepare scores and BPE data.
* `N`: Number of hypotheses per each source sentence. We use 50 in the paper.
* `SPLIT`: Name of the data split, i.e. train, valid, test. Use split_name, split_name1, split_name2, ..., if there are multiple datasets for a split, e.g. train, train1, valid, valid1.
* `NUM_SHARDS`: Number of shards. Set this to 1 for non-train splits.
* `METRIC`: The metric for DrNMT to optimize for. We support either `bleu` or `ter`.
```
# For each data split, e.g. train, valid, test, etc., run the following:
SOURCE_FILE=/path/to/source_sentence_file
TARGET_FILE=/path/to/target_sentence_file
HYPO_FILE=/path/to/hypo_file
XLMR_DIR=/path/to/xlmr
OUTPUT_DIR=/path/to/output
python scripts/prep_data.py \
--input-source ${SOURCE_FILE} \
--input-target ${TARGET_FILE} \
--input-hypo ${HYPO_FILE} \
--output-dir ${OUTPUT_DIR} \
--split $SPLIT
--beam $N \
--sentencepiece-model ${XLMR_DIR}/sentencepiece.bpe.model \
--metric $METRIC \
--num-shards ${NUM_SHARDS}
# The script will create ${OUTPUT_DIR}/$METRIC with ${NUM_SHARDS} splits.
# Under split*/input_src, split*/input_tgt and split*/$METRIC, there will be $SPLIT.bpe and $SPLIT.$METRIC files, respectively.
```
4. Pre-process the data into fairseq format.
```
# use comma to separate if there are more than one train or valid set
for suffix in src tgt ; do
fairseq-preprocess --only-source \
--trainpref ${OUTPUT_DIR}/$METRIC/split1/input_${suffix}/train.bpe \
--validpref ${OUTPUT_DIR}/$METRIC/split1/input_${suffix}/valid.bpe \
--destdir ${OUTPUT_DIR}/$METRIC/split1/input_${suffix} \
--workers 60 \
--srcdict ${XLMR_DIR}/dict.txt
done
for i in `seq 2 ${NUM_SHARDS}`; do
for suffix in src tgt ; do
fairseq-preprocess --only-source \
--trainpref ${OUTPUT_DIR}/$METRIC/split${i}/input_${suffix}/train.bpe \
--destdir ${OUTPUT_DIR}/$METRIC/split${i}/input_${suffix} \
--workers 60 \
--srcdict ${XLMR_DIR}/dict.txt
ln -s ${OUTPUT_DIR}/$METRIC/split1/input_${suffix}/valid* ${OUTPUT_DIR}/$METRIC/split${i}/input_${suffix}/.
done
ln -s ${OUTPUT_DIR}/$METRIC/split1/$METRIC/valid* ${OUTPUT_DIR}/$METRIC/split${i}/$METRIC/.
done
```
## Training
```
EXP_DIR=/path/to/exp
# An example of training the model with the config for De-En experiment in the paper.
# The config uses 16 GPUs and 50 hypotheses.
# For training with fewer number of GPUs, set
# distributed_training.distributed_world_size=k +optimization.update_freq='[x]' where x = 16/k
# For training with fewer number of hypotheses, set
# task.mt_beam=N dataset.batch_size=N dataset.required_batch_size_multiple=N
fairseq-hydra-train -m \
--config-dir config/ --config-name deen \
task.data=${OUTPUT_DIR}/$METRIC/split1/ \
task.num_data_splits=${NUM_SHARDS} \
model.pretrained_model=${XLMR_DIR}/model.pt \
common.user_dir=${FAIRSEQ_ROOT}/examples/discriminative_reranking_nmt \
checkpoint.save_dir=${EXP_DIR}
```
## Inference & scoring
Perform DrNMT reranking (fw + reranker score)
1. Tune weights on valid sets.
```
# genrate N hypotheses with the base MT model (fw score)
VALID_SOURCE_FILE=/path/to/source_sentences # one sentence per line, converted to the sentencepiece used by the base MT model
VALID_TARGET_FILE=/path/to/target_sentences # one sentence per line in raw text, i.e. no sentencepiece and tokenization
MT_MODEL=/path/to/mt_model
MT_DATA_PATH=/path/to/mt_data
cat ${VALID_SOURCE_FILE} | \
fairseq-interactive ${MT_DATA_PATH} \
--max-tokens 4000 --buffer-size 16 \
--num-workers 32 --path ${MT_MODEL} \
--beam $N --nbest $N \
--post-process sentencepiece &> valid-hypo.out
# replace "bleu" with "ter" to optimize for TER
python drnmt_rerank.py \
${OUTPUT_DIR}/$METRIC/split1/ \
--path ${EXP_DIR}/checkpoint_best.pt \
--in-text valid-hypo.out \
--results-path ${EXP_DIR} \
--gen-subset valid \
--target-text ${VALID_TARGET_FILE} \
--user-dir ${FAIRSEQ_ROOT}/examples/discriminative_reranking_nmt \
--bpe sentencepiece \
--sentencepiece-model ${XLMR_DIR}/sentencepiece.bpe.model \
--beam $N \
--batch-size $N \
--metric bleu \
--tune
```
2. Apply best weights on test sets
```
# genrate N hypotheses with the base MT model (fw score)
TEST_SOURCE_FILE=/path/to/source_sentences # one sentence per line, converted to the sentencepiece used by the base MT model
cat ${TEST_SOURCE_FILE} | \
fairseq-interactive ${MT_DATA_PATH} \
--max-tokens 4000 --buffer-size 16 \
--num-workers 32 --path ${MT_MODEL} \
--beam $N --nbest $N \
--post-process sentencepiece &> test-hypo.out
# replace "bleu" with "ter" to evaluate TER
# Add --target-text for evaluating BLEU/TER,
# otherwise the script will only generate the hypotheses with the highest scores only.
python drnmt_rerank.py \
${OUTPUT_DIR}/$METRIC/split1/ \
--path ${EXP_DIR}/checkpoint_best.pt \
--in-text test-hypo.out \
--results-path ${EXP_DIR} \
--gen-subset test \
--user-dir ${FAIRSEQ_ROOT}/examples/discriminative_reranking_nmt \
--bpe sentencepiece \
--sentencepiece-model ${XLMR_DIR}/sentencepiece.bpe.model \
--beam $N \
--batch-size $N \
--metric bleu \
--fw-weight ${BEST_FW_WEIGHT} \
--lenpen ${BEST_LENPEN}
```
## Citation
```bibtex
@inproceedings{lee2021discriminative,
title={Discriminative Reranking for Neural Machine Translation},
author={Lee, Ann and Auli, Michael and Ranzato, Marc'Aurelio},
booktitle={ACL},
year={2021}
}
```
|