model.py 6.1 KB
Newer Older
Y
Yibing Liu 已提交
1
import six
Y
Yibing Liu 已提交
2 3 4 5 6 7 8
import numpy as np
import paddle.fluid as fluid
import utils.layers as layers


class Net(object):
    def __init__(self, max_turn_num, max_turn_len, vocab_size, emb_size,
Y
Yibing Liu 已提交
9
                 stack_num, channel1_num, channel2_num):
Y
Yibing Liu 已提交
10 11 12 13 14 15

        self._max_turn_num = max_turn_num
        self._max_turn_len = max_turn_len
        self._vocab_size = vocab_size
        self._emb_size = emb_size
        self._stack_num = stack_num
Y
Yibing Liu 已提交
16 17
        self._channel1_num = channel1_num
        self._channel2_num = channel2_num
Y
Yibing Liu 已提交
18
        self.word_emb_name = "shared_word_emb"
S
sneaxiy 已提交
19 20 21
        self.use_stack_op = True
        self.use_mask_cache = True
        self.use_sparse_embedding = True
Y
Yibing Liu 已提交
22

Y
Yibing Liu 已提交
23 24 25 26 27
    def set_word_embedding(self, word_emb, place):
        word_emb_param = fluid.global_scope().find_var(
            self.word_emb_name).get_tensor()
        word_emb_param.set(word_emb, place)

Y
Yibing Liu 已提交
28
    def create_network(self):
S
sneaxiy 已提交
29 30
        mask_cache = dict() if self.use_mask_cache else None

Y
Yibing Liu 已提交
31
        turns_data = []
Y
Yibing Liu 已提交
32
        for i in six.moves.xrange(self._max_turn_num):
Y
Yibing Liu 已提交
33 34 35 36 37 38 39
            turn = fluid.layers.data(
                name="turn_%d" % i,
                shape=[self._max_turn_len, 1],
                dtype="int32")
            turns_data.append(turn)

        turns_mask = []
Y
Yibing Liu 已提交
40
        for i in six.moves.xrange(self._max_turn_num):
Y
Yibing Liu 已提交
41 42
            turn_mask = fluid.layers.data(
                name="turn_mask_%d" % i,
S
sneaxiy 已提交
43
                shape=[self._max_turn_len, 1],
Y
Yibing Liu 已提交
44 45 46 47 48 49
                dtype="float32")
            turns_mask.append(turn_mask)

        response = fluid.layers.data(
            name="response", shape=[self._max_turn_len, 1], dtype="int32")
        response_mask = fluid.layers.data(
S
sneaxiy 已提交
50 51 52
            name="response_mask",
            shape=[self._max_turn_len, 1],
            dtype="float32")
Y
Yibing Liu 已提交
53 54 55 56 57
        label = fluid.layers.data(name="label", shape=[1], dtype="float32")

        response_emb = fluid.layers.embedding(
            input=response,
            size=[self._vocab_size + 1, self._emb_size],
S
sneaxiy 已提交
58
            is_sparse=self.use_sparse_embedding,
Y
Yibing Liu 已提交
59 60 61 62 63 64 65 66
            param_attr=fluid.ParamAttr(
                name=self.word_emb_name,
                initializer=fluid.initializer.Normal(scale=0.1)))

        # response part
        Hr = response_emb
        Hr_stack = [Hr]

Y
Yibing Liu 已提交
67
        for index in six.moves.xrange(self._stack_num):
Y
Yibing Liu 已提交
68 69 70 71 72 73 74
            Hr = layers.block(
                name="response_self_stack" + str(index),
                query=Hr,
                key=Hr,
                value=Hr,
                d_key=self._emb_size,
                q_mask=response_mask,
S
sneaxiy 已提交
75 76
                k_mask=response_mask,
                mask_cache=mask_cache)
Y
Yibing Liu 已提交
77 78 79 80
            Hr_stack.append(Hr)

        # context part
        sim_turns = []
Y
Yibing Liu 已提交
81
        for t in six.moves.xrange(self._max_turn_num):
Y
Yibing Liu 已提交
82 83 84
            Hu = fluid.layers.embedding(
                input=turns_data[t],
                size=[self._vocab_size + 1, self._emb_size],
S
sneaxiy 已提交
85
                is_sparse=self.use_sparse_embedding,
Y
Yibing Liu 已提交
86 87 88 89 90
                param_attr=fluid.ParamAttr(
                    name=self.word_emb_name,
                    initializer=fluid.initializer.Normal(scale=0.1)))
            Hu_stack = [Hu]

Y
Yibing Liu 已提交
91
            for index in six.moves.xrange(self._stack_num):
Y
Yibing Liu 已提交
92 93 94 95 96 97 98 99
                # share parameters
                Hu = layers.block(
                    name="turn_self_stack" + str(index),
                    query=Hu,
                    key=Hu,
                    value=Hu,
                    d_key=self._emb_size,
                    q_mask=turns_mask[t],
S
sneaxiy 已提交
100 101
                    k_mask=turns_mask[t],
                    mask_cache=mask_cache)
Y
Yibing Liu 已提交
102 103 104 105 106
                Hu_stack.append(Hu)

            # cross attention 
            r_a_t_stack = []
            t_a_r_stack = []
Y
Yibing Liu 已提交
107
            for index in six.moves.xrange(self._stack_num + 1):
Y
Yibing Liu 已提交
108 109 110 111 112 113 114
                t_a_r = layers.block(
                    name="t_attend_r_" + str(index),
                    query=Hu_stack[index],
                    key=Hr_stack[index],
                    value=Hr_stack[index],
                    d_key=self._emb_size,
                    q_mask=turns_mask[t],
S
sneaxiy 已提交
115 116
                    k_mask=response_mask,
                    mask_cache=mask_cache)
Y
Yibing Liu 已提交
117 118 119 120 121 122 123
                r_a_t = layers.block(
                    name="r_attend_t_" + str(index),
                    query=Hr_stack[index],
                    key=Hu_stack[index],
                    value=Hu_stack[index],
                    d_key=self._emb_size,
                    q_mask=response_mask,
S
sneaxiy 已提交
124 125
                    k_mask=turns_mask[t],
                    mask_cache=mask_cache)
Y
Yibing Liu 已提交
126 127 128 129 130 131 132

                t_a_r_stack.append(t_a_r)
                r_a_t_stack.append(r_a_t)

            t_a_r_stack.extend(Hu_stack)
            r_a_t_stack.extend(Hr_stack)

S
sneaxiy 已提交
133 134 135 136
            if self.use_stack_op:
                t_a_r = fluid.layers.stack(t_a_r_stack, axis=1)
                r_a_t = fluid.layers.stack(r_a_t_stack, axis=1)
            else:
Y
Yibing Liu 已提交
137
                for index in six.moves.xrange(len(t_a_r_stack)):
S
sneaxiy 已提交
138 139 140 141
                    t_a_r_stack[index] = fluid.layers.unsqueeze(
                        input=t_a_r_stack[index], axes=[1])
                    r_a_t_stack[index] = fluid.layers.unsqueeze(
                        input=r_a_t_stack[index], axes=[1])
Y
Yibing Liu 已提交
142

S
sneaxiy 已提交
143 144
                t_a_r = fluid.layers.concat(input=t_a_r_stack, axis=1)
                r_a_t = fluid.layers.concat(input=r_a_t_stack, axis=1)
Y
Yibing Liu 已提交
145

Y
Yibing Liu 已提交
146
            # sim shape: [batch_size, 2*(stack_num+1), max_turn_len, max_turn_len]    
S
sneaxiy 已提交
147 148
            sim = fluid.layers.matmul(
                x=t_a_r, y=r_a_t, transpose_y=True, alpha=1 / np.sqrt(200.0))
Y
Yibing Liu 已提交
149 150
            sim_turns.append(sim)

S
sneaxiy 已提交
151 152 153
        if self.use_stack_op:
            sim = fluid.layers.stack(sim_turns, axis=2)
        else:
Y
Yibing Liu 已提交
154
            for index in six.moves.xrange(len(sim_turns)):
S
sneaxiy 已提交
155 156
                sim_turns[index] = fluid.layers.unsqueeze(
                    input=sim_turns[index], axes=[2])
Y
Yibing Liu 已提交
157
            # sim shape: [batch_size, 2*(stack_num+1), max_turn_num, max_turn_len, max_turn_len]
S
sneaxiy 已提交
158
            sim = fluid.layers.concat(input=sim_turns, axis=2)
Y
Yibing Liu 已提交
159

Y
Yibing Liu 已提交
160
        final_info = layers.cnn_3d(sim, self._channel1_num, self._channel2_num)
Y
Yibing Liu 已提交
161 162
        loss, logits = layers.loss(final_info, label)
        return loss, logits