There is a task in NLP to retrieve the context closest semantically or the most suitable response to a given context from some context (response) database. The code in this repository uses a deep learning approach to address the question answer selection task. Currently, a basic model is implemented with bidirectional long short-term memory (biLSTM), with max pooling and without attention. The model is applied to the InsuranceQA dataset https://github.com/shuzi/insuranceQA .
The distinguishing feature of the model is the use of triplet loss [1, 2]. This loss has a margin hyperparameter, which usually ranges from 0.01 to 0.2. It is required to provide positive and negative response candidates for each context from the dataset to train the model. Sampling of negative candidates can be performed globaly from the whole response set or from the pools of responses predefined separatly for each context. The same is true for validation and test, i.e. validation and test can be carried out for the entire set of answers or for the answers selected separately for each context. There is a possibility in the model to encode contexts and responses with biLSTM layers having shared or separate weights.
Each train data sample for feeding the model is arranged as follows:
{'context': [21507, 4696, 5843, 13035, 1628, 20923, 16083, 18690], 'response': 7009, 'pos_pool': [7009, 7010], 'neg_pool': None}
The context has a key "context" in the data sample. It is represented by a list of integers which are keys that give the list of tokens using the dictionary "token–integer". The correct response has the key "response", its value is always a single integer. The list of possible correct responses (there may be several) can be obtained with the key "pos_pool". The "response" value should be equal to one item from the list obtained using the key "pos_pool". The list of possible negative responses (there can be a lot of them, 100–10000) is represented by the key "neg_pool". Its value is None, when global sampling is used, or the list of fixed length, when sampling from predefined negative responses is used. It is important that values in "pos_pool" and "negative_pool" did not overlap. Single responses in "response", "pos_pool", "neg_pool" are represented by single integers that give lists of integers using the dictionary "integer–list of integers". These lists of integers can be converted to lists of tokens with the same dictionary "integer–token" which is used for contexts. The additional "integer–list of integers" vocabulary is used to not store in the form of sequences all possible negative responses. Validation and test data samples are almost the same as train samples. The difference is that candidates for ranking are taken from "neg_pool" list of the length 500.
To use the pre-trained model for inference one should run the following command:
python -m deeppavlov.deep interact deeppavlov/configs/ranking/insurance_config.json
Now user can enter a text of context and get relevant contexts and responses:
:: how much to pay for auto insurance?
>> {'contexts': ['how much can I expect pay for auto insurance', 'how much will insurance pay for my total car', 'how much can I expect pay in car insurance'], 'responses': ['the cost of auto insurance be based on several factor include your driving record , claim history , type of vehicle , credit score where you live and how far you travel to and from work I will recommend work with an independent agent who can shop several company find the good policy for you', 'there be not any absolute answer to this question rate for auto insurance coverage can vary greatly from carrier to carrier and from area to area contact local agent in your area find out about coverage availablity and pricing within your area look for an agent that you be comfortable working with as they will be the first last point of contact in most instance', 'the cost of auto insurance coverage for any vehicle or driver can vary greatly thing that effect your auto insurance rate be geographical location , vehicle , age (s) of driver (s) , type of coverage desire , motor vehicle record of all driver , credit rating of all driver and more contact a local agent get a quote a quote cost nothing but will let you know where your rate will']}
To train the model on the InsuranceQA dataset one should run the command:
python -m deeppavlov.deep interact deeppavlov/configs/ranking/insurance_config.json
All parameters from insurance_config.json config file are described in the table below.
Parameter | Description |
---|---|
dataset_reader | reads datasets from files. |
name | str, a registered name of the dataset reader. |
data_path | str, a directory where data files are stored. |
dataset_iterator | provides models with data. |
name | str, a registered name of the dataset. |
seed | int or None (default=None), a seed for a batch generator. |
sample_candiates | {"global", "pool"}, a method of negative sampling in train data. If "pool", negative candidates for each data sample should be provided. If "global", negative sampling over the whole data will be performed. |
sample_candiates_valid | {"global", "pool"}, a method of selecting_candidates for ranking in valid data. If "pool", candidates for ranking for each data sample should be provided. If "global", all data samples will be taken as candidates for ranking. |
sample_candiates_test | {"global", "pool"}, a method of selecting_candidates for ranking in valid data. If "pool", candidates for ranking for each data sample should be provided. If "global", all data samples will be taken as candidates for ranking. |
num_negative_samples | int, a number of negative samples to use if "sample_candiates" is set to "pool". |
num_ranking_samples_valid | int, a number of negative samples to use if "sample_candiates_valid" is set to "pool". |
num_ranking_samples_test | int, a number of negative samples to use if "sample_candiates_test" is set to "pool". |
chainer | pipeline from heterogeneous components. |
in | list of str, a user-defined list of input names, i.e ["x"], ["x0", "x1"]. |
in_y | list of str, a user-defined list of input target names , i.e ["y"], ["y0", "y1"]. |
out | list of str, a user-defined list of output names, i.e. ["y_pred"], ["y_pred0", "y_pred1"]. |
pipe | contains the sequence of model components (including vocabs, preprocessors, main components, postprocessors etc.). |
parameters of the main part of a model | |
in | the same as "in" parameter in the "chainer". |
in_y | the same as "in_y" parameter in the "chainer". |
out | the same as "out" parameter in the "chainer". |
name | str, a registered name of the model. |
device_num | int, a GPU card number to train the model on, if several cards are available in a system. |
load_path | str, a path to a file from which model files will be loaded. |
save_path | str, a path to a file where model files will be saved. |
train_now | bool, if it is True, than the model training will be done, else validation and test only. |
vocabs_path | str, a path to a directory with data files from where the model vocabularies will be built. |
download_url | str, a URL where a pre-trained model with word embeddings is stored. |
embeddings | {"wor2vec", "fasttext"}, a type of the pre-trained embeddings model. |
seed | int or None (default=None), a seed to initialize the model weights. |
max_sequence_length | int, a maximum number of tokens in an input sequence. If the sequence is shorter than the "max_sequence_length" it will be padded with a default token, otherwise the sequence will be truncated. |
padding | {"pre", "post"}, pad either before or after each sequence if it is shorter than "max_sequence_length". |
truncating | {"pre", "post"}, remove values from sequences larger than "max_sequence_length", either at the beginning or at the end of the sequences. |
reccurent | {"lstm", "bilstm"}, a type of a reccurent neural network (LSTM or bi-LSTM) to encode an input sequence. |
max_pooling | bool, if it is True the max-pooling operation will be performed, else the last hidden state from the reccurent neural network will be taken. |
type_of_weights | {"shared", "separate"}, use shared of separate weights to encode the context and response. |
hidden_dim | int, a size of a hidden state if the "reccurent" parameter is set to the "lstm" or the half-size if the "reccurent" is set to the "bilstm". |
learning_rate | float, learning rate for training. |
margin | float, a margin to use in a triplet loss. |
load_path | str, a path to a file from which model files will be loaded. |
save_path | str, a path to a file where model files will be saved. |
interact_pred_num | int, first "interact_pred_num" best candidates for context and response to show in the "interact" regime. |
train | parameters for training |
epochs | int, a number of epochs for training. |
batch_size | int, a batch size for training. |
metrics | a list of metrics names , top-1 recall "r@1", "r@2", "r@5" and the average position of the correct response among all response candidates "rank_response" are available for the model. |
validation_patience | int, for how many epochs the training can continue without improvement of the metric value on the validation set. |
val_every_n_epochs | int, a frequency of validation during training (validate every n epochs). |
Model | Validation | Test1 |
---|---|---|
Architecture II: (HLQA(200) CNNQA(4000) 1-MaxPooling Tanh) [1] | 61.8 | 62.8 |
QA-LSTM basic-model(max pooling) [2] | 64.3 | 63.1 |
Our model (biLSTM, max pooling) | 63.5 | 62.2 |
[1] Feng, Minwei, et al. "Applying deep learning to answer selection: A study and an open task." Automatic Speech Recognition and Understanding (ASRU), 2015 IEEE Workshop on. IEEE, 2015.
[2] Tan, Ming, et al. "LSTM-based deep learning models for non-factoid answer selection." arXiv preprint arXiv:1511.04108 (2015).