single_trainer.py 4.8 KB
Newer Older
T
tangwei 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
# Copyright (c) 2020 PaddlePaddle Authors. All Rights Reserved.
#
# 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.
"""
Training use fluid with one node only.
"""

from __future__ import print_function
T
tangwei 已提交
19

T
tangwei 已提交
20
import time
T
tangwei 已提交
21
import logging
T
tangwei 已提交
22

T
tangwei 已提交
23 24
import paddle.fluid as fluid

25 26
from paddlerec.core.trainers.transpiler_trainer import TranspileTrainer
from paddlerec.core.utils import envs
T
tangwei 已提交
27 28 29 30 31 32

logging.basicConfig(format="%(asctime)s - %(levelname)s - %(message)s")
logger = logging.getLogger("fluid")
logger.setLevel(logging.INFO)


T
tangwei 已提交
33
class SingleTrainer(TranspileTrainer):
T
tangwei 已提交
34 35 36
    def processor_register(self):
        self.regist_context_processor('uninit', self.instance)
        self.regist_context_processor('init_pass', self.init)
C
chengmo 已提交
37
        self.regist_context_processor('startup_pass', self.startup)
T
tangwei 已提交
38 39 40

        if envs.get_platform() == "LINUX" and envs.get_global_env(
                "dataset_class", None, "train.reader") != "DataLoader":
T
tangwei 已提交
41 42 43 44
            self.regist_context_processor('train_pass', self.dataset_train)
        else:
            self.regist_context_processor('train_pass', self.dataloader_train)

T
tangwei 已提交
45 46 47 48
        self.regist_context_processor('infer_pass', self.infer)
        self.regist_context_processor('terminal_pass', self.terminal)

    def init(self, context):
T
tangwei 已提交
49
        self.model.train_net()
T
tangwei 已提交
50
        optimizer = self.model.optimizer()
T
tangwei 已提交
51
        optimizer.minimize((self.model.get_avg_cost()))
T
tangwei 已提交
52 53 54 55

        self.fetch_vars = []
        self.fetch_alias = []
        self.fetch_period = self.model.get_fetch_period()
T
tangwei 已提交
56

T
tangwei 已提交
57 58 59 60
        metrics = self.model.get_metrics()
        if metrics:
            self.fetch_vars = metrics.values()
            self.fetch_alias = metrics.keys()
C
chengmo 已提交
61 62 63 64 65 66
        evaluate_only = envs.get_global_env(
            'evaluate_only', False, namespace='evaluate')
        if evaluate_only:
            context['status'] = 'infer_pass'
        else:
            context['status'] = 'startup_pass'
C
chengmo 已提交
67 68 69

    def startup(self, context):
        self._exe.run(fluid.default_startup_program())
T
tangwei 已提交
70 71
        context['status'] = 'train_pass'

T
tangwei 已提交
72
    def dataloader_train(self, context):
M
malin10 已提交
73
        reader = self._get_dataloader("TRAIN")
T
tangwei 已提交
74
        epochs = envs.get_global_env("train.epochs")
T
tangwei 已提交
75

T
tangwei 已提交
76 77
        program = fluid.compiler.CompiledProgram(fluid.default_main_program(
        )).with_data_parallel(loss_name=self.model.get_avg_cost().name)
T
tangwei 已提交
78 79 80 81 82 83 84 85

        metrics_varnames = []
        metrics_format = []

        metrics_format.append("{}: {{}}".format("epoch"))
        metrics_format.append("{}: {{}}".format("batch"))

        for name, var in self.model.get_metrics().items():
T
tangwei 已提交
86
            metrics_varnames.append(var.name)
T
tangwei 已提交
87 88 89
            metrics_format.append("{}: {{}}".format(name))

        metrics_format = ", ".join(metrics_format)
T
tangwei 已提交
90

T
tangwei 已提交
91 92 93 94 95
        for epoch in range(epochs):
            reader.start()
            batch_id = 0
            try:
                while True:
T
tangwei 已提交
96 97
                    metrics_rets = self._exe.run(program=program,
                                                 fetch_list=metrics_varnames)
T
tangwei 已提交
98 99

                    metrics = [epoch, batch_id]
T
tangwei 已提交
100
                    metrics.extend(metrics_rets)
T
tangwei 已提交
101

M
malin10 已提交
102
                    if batch_id % self.fetch_period == 0 and batch_id != 0:
T
tangwei 已提交
103
                        print(metrics_format.format(*metrics))
T
tangwei 已提交
104 105 106
                    batch_id += 1
            except fluid.core.EOFException:
                reader.reset()
M
malin10 已提交
107
            self.save(epoch, "train", is_fleet=False)
T
tangwei 已提交
108 109 110 111

        context['status'] = 'infer_pass'

    def dataset_train(self, context):
M
malin10 已提交
112
        dataset = self._get_dataset("TRAIN")
T
tangwei 已提交
113
        ins = self._get_dataset_ins()
T
tangwei 已提交
114

T
tangwei 已提交
115
        epochs = envs.get_global_env("train.epochs")
T
tangwei 已提交
116
        for i in range(epochs):
T
tangwei 已提交
117
            begin_time = time.time()
T
tangwei 已提交
118 119 120 121 122 123
            self._exe.train_from_dataset(
                program=fluid.default_main_program(),
                dataset=dataset,
                fetch_list=self.fetch_vars,
                fetch_info=self.fetch_alias,
                print_period=self.fetch_period)
T
tangwei 已提交
124
            end_time = time.time()
T
for mat  
tangwei 已提交
125
            times = end_time - begin_time
T
tangwei 已提交
126 127
            print("epoch {} using time {}, speed {:.2f} lines/s".format(
                i, times, ins / times))
T
tangwei 已提交
128

T
tangwei 已提交
129 130 131 132 133 134 135
            self.save(i, "train", is_fleet=False)
        context['status'] = 'infer_pass'

    def terminal(self, context):
        for model in self.increment_models:
            print("epoch :{}, dir: {}".format(model[0], model[1]))
        context['is_exit'] = True