/* Copyright (c) 2016 PaddlePaddle Authors. All Rights Reserve. 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. */ #include "paddle/gserver/evaluators/Evaluator.h" #include "paddle/gserver/gradientmachines/NeuralNetwork.h" #include "paddle/utils/Stat.h" #include "paddle/utils/StringUtil.h" DECLARE_int32(trainer_id); namespace paddle { void Evaluator::eval(const NeuralNetwork& nn) { std::vector arguments; arguments.reserve(config_.input_layers_size()); for (const std::string& name : config_.input_layers()) { arguments.push_back(nn.getLayer(name)->getOutput()); } SetDevice device(arguments[0].deviceId); real score = evalImp(arguments); totalScore_ += score; updateSamplesNum(arguments); } /** * @brief classification error Evaluator * * The config file api is classification_error_evaluator. */ class ClassificationErrorEvaluator : public Evaluator { public: virtual void updateSamplesNum(const std::vector& arguments) { if (3 == arguments.size()) { numSamples_ += arguments[2].value->getSum(); } else { numSamples_ += arguments[0].getBatchSize(); } } MatrixPtr calcError(std::vector& arguments) { CHECK_GE(arguments.size(), (size_t)2); CHECK_LE(arguments.size(), (size_t)3); MatrixPtr& output = arguments[0].value; IVectorPtr& label = arguments[1].ids; MatrixPtr& multiBinaryLabel = arguments[1].value; // For multi binary label bool supportWeight = (3 == arguments.size()) ? true : false; MatrixPtr weight = supportWeight ? arguments[2].value : nullptr; if (nullptr == output || (nullptr == label && nullptr == multiBinaryLabel) || (supportWeight && nullptr == weight)) { return 0; } if (label != nullptr) { CHECK_EQ(label->getSize(), output->getHeight()); } else { CHECK_EQ(multiBinaryLabel->getHeight(), output->getHeight()); CHECK_EQ(multiBinaryLabel->getWidth(), output->getWidth()); } if (supportWeight) { CHECK_EQ(output->getHeight(), weight->getHeight()); CHECK_EQ((size_t)1, weight->getWidth()); } const MatrixPtr errorMat = Matrix::create(output->getHeight(), 1, /* trans= */ false, useGpu(arguments[0].deviceId)); errorMat->zeroMem(); if (label != nullptr) { errorMat->classificationError(*output, *label); } else if (dynamic_cast(multiBinaryLabel.get()) || dynamic_cast(multiBinaryLabel.get())) { errorMat->classificationErrorMulti( *output, *multiBinaryLabel, config_.classification_threshold()); } else { errorMat->binaryClassificationError( 0, *output, *multiBinaryLabel, config_.classification_threshold()); } if (supportWeight) { errorMat->dotMul(*errorMat, *weight); } return errorMat; } virtual real evalImp(std::vector& arguments) { MatrixPtr errorMat = calcError(arguments); return errorMat->getSum(); } virtual void distributeEval(ParameterClient2* client) { mergeResultsOfAllClients(client); } }; /** * @brief sequence classification error Evaluator * @note sequence level classification error stats, * if any frame in one sequence has error, the sequence is error */ class SequenceClassificationErrorEvaluator : public ClassificationErrorEvaluator { public: virtual void updateSamplesNum(const std::vector& arguments) { numSamples_ += arguments[0].getNumSequences(); } virtual real evalImp(std::vector& arguments) { auto sequenceStartPositions = arguments[0].sequenceStartPositions->getVector(false); CHECK(sequenceStartPositions != nullptr); const int* starts = sequenceStartPositions->getData(); MatrixPtr errorMat = calcError(arguments); int errCounter = 0; CpuVector errorVec(0, nullptr); for (size_t i = 0; i < sequenceStartPositions->getSize() - 1; ++i) { errorVec.subVecFrom( errorMat->getData(), starts[i], starts[i + 1] - starts[i]); if (errorVec.getSum() > 0) { errCounter += 1; } } return static_cast(errCounter); } virtual void distributeEval(ParameterClient2* client) { mergeResultsOfAllClients(client); } }; REGISTER_EVALUATOR(seq_classification_error, SequenceClassificationErrorEvaluator); /** * @brief sum Evaluator * Calculate the sum of output or label * * The config file api is sum_evaluator. */ class SumEvaluator : public Evaluator { public: SumEvaluator() : cpuLabel_(nullptr), cpuWeight_(nullptr) {} virtual void updateSamplesNum(const std::vector& arguments) { if (2 == arguments.size()) { numSamples_ += arguments[1].value->getSum(); } else { numSamples_ += arguments[0].getBatchSize(); } } virtual real evalImp(std::vector& arguments) { REGISTER_TIMER("SumEvaluator"); CHECK_GE(arguments.size(), (size_t)1); CHECK_LE(arguments.size(), (size_t)2); bool supportWeight = (2 == arguments.size()) ? true : false; if (supportWeight) { if (nullptr == arguments[1].value) { return 0; } CHECK_EQ(arguments[1].value->getWidth(), (size_t)1); } // The sum of output if (arguments[0].value) { if (supportWeight) { CHECK_EQ(arguments[0].value->getHeight(), arguments[1].value->getHeight()); MatrixPtr tmpMat = Matrix::create(arguments[0].value->getHeight(), arguments[0].value->getWidth(), /* trans= */ false, arguments[0].value->useGpu()); tmpMat->copyFrom(*arguments[0].value); tmpMat->rowScale(0, *tmpMat, *arguments[1].value); return tmpMat->getSum(); } else { return arguments[0].value->getSum(); } // The sum of label } else if (arguments[0].ids) { size_t insNum = arguments[0].ids->getSize(); IVectorPtr label = arguments[0].ids; MatrixPtr weight = supportWeight ? arguments[1].value : nullptr; if (dynamic_cast(label.get())) { IVector::resizeOrCreate(cpuLabel_, insNum, false); cpuLabel_->copyFrom(*arguments[0].ids); if (supportWeight) { CHECK_EQ(insNum, arguments[1].value->getHeight()); Matrix::resizeOrCreate(cpuWeight_, insNum, (size_t)1, false, false); cpuWeight_->copyFrom(*arguments[1].value); } label = cpuLabel_; weight = cpuWeight_; } if (supportWeight) { real score = 0.0; int* labelD = label->getData(); real* weightD = weight->getData(); for (size_t i = 0; i < insNum; ++i) { score += (labelD[i] * weightD[i]); } return score; } else { return label->getSum(); } } else { return 0; } } virtual void distributeEval(ParameterClient2* client) { mergeResultsOfAllClients(client); } private: IVectorPtr cpuLabel_; MatrixPtr cpuWeight_; }; /** * @brief column sum Evaluator * @note column sum for the colIdx-th column * * - colIdx = 0: the 0-th column. * - colIdx > 0: the colIdx-th column. * - colIdx < 0: the last colIdx-th column. * * The config file api is column_sum_evaluator. * */ class ColumnSumEvaluator : public Evaluator { public: explicit ColumnSumEvaluator(int32_t colIdx) : colIdx_(colIdx), colNum_(0), sum_(nullptr) {} virtual void start() { Evaluator::start(); if (nullptr != sum_) { sum_->zeroMem(); } } virtual void updateSamplesNum(const std::vector& arguments) { if (2 == arguments.size()) { numSamples_ += arguments[1].value->getSum(); } else { numSamples_ += arguments[0].getBatchSize(); } } virtual real evalImp(std::vector& arguments) { REGISTER_TIMER("ColumnSumEvaluator"); CHECK_GE(arguments.size(), (size_t)1); CHECK_LE(arguments.size(), (size_t)2); bool supportWeight = (2 == arguments.size()) ? true : false; if (nullptr == arguments[0].value || (supportWeight && nullptr == arguments[1].value)) { return 0; } size_t insNum = arguments[0].value->getHeight(); size_t colNum = arguments[0].value->getWidth(); if (nullptr == sum_) { sum_ = Matrix::create((size_t)1, colNum, false, /* useGpu */ false); colNum_ = colNum; sum_->zeroMem(); } else { CHECK_EQ(colNum, sum_->getWidth()); } if (supportWeight) { CHECK_EQ(insNum, arguments[1].value->getHeight()); CHECK_EQ((size_t)1, arguments[1].value->getWidth()); MatrixPtr tmpMat = Matrix::create(insNum, colNum); if (arguments[0].value->useGpu()) { tmpMat->copyFrom(*arguments[0].value); } if (!arguments[1].value->useGpu()) { if (!arguments[0].value->useGpu()) { tmpMat->rowScale(0, *arguments[0].value, *arguments[1].value); } else { tmpMat->rowScale(0, *tmpMat, *arguments[1].value); } } else { MatrixPtr tmp2 = Matrix::create(insNum, 1); tmp2->copyFrom(*arguments[1].value); if (!arguments[0].value->useGpu()) { tmpMat->rowScale(0, *arguments[0].value, *tmp2); } else { tmpMat->rowScale(0, *tmpMat, *tmp2); } } sum_->accumulateColSum(*tmpMat); } else { if (!arguments[0].value->useGpu()) { sum_->accumulateColSum(*arguments[0].value); } else { MatrixPtr tmpMat = Matrix::create(insNum, colNum); tmpMat->copyFrom(*arguments[0].value); sum_->accumulateColSum(*tmpMat); } } return 0; } virtual void printStats(std::ostream& os) const { CHECK(colIdx_ + (int32_t)colNum_ >= 0 && colIdx_ - (int32_t)colNum_ < 0) << "column index [" << colIdx_ << "] out of range [-" << colNum_ << ", " << colNum_ << ")"; size_t colIdx = 0; if (colIdx_ >= 0) { colIdx = colIdx_; } else { colIdx = colNum_ + colIdx_; } os << config_.name() << "=" << (numSamples_ ? sum_->getElement(0, colIdx) / numSamples_ : 0); } void distributeEval(ParameterClient2* client) { client->reduce( sum_->getData(), sum_->getData(), colNum_, FLAGS_trainer_id, 0); client->reduce(&numSamples_, &numSamples_, 1, FLAGS_trainer_id, 0); } private: ColumnSumEvaluator() {} int32_t colIdx_; size_t colNum_; MatrixPtr sum_; /* cpu matrix */ }; void AucEvaluator::start() { Evaluator::start(); memset(statPos_, 0, sizeof(statPos_)); memset(statNeg_, 0, sizeof(statNeg_)); } real AucEvaluator::evalImp(std::vector& arguments) { REGISTER_TIMER("AucEvaluator"); CHECK_GE(arguments.size(), (size_t)2); CHECK_LE(arguments.size(), (size_t)3); MatrixPtr output = arguments[0].value; IVectorPtr label = arguments[1].ids; bool supportWeight = (3 == arguments.size()) ? true : false; MatrixPtr weight = supportWeight ? arguments[2].value : nullptr; if (nullptr == output || nullptr == label || (supportWeight && nullptr == weight)) { return 0; } size_t insNum = output->getHeight(); size_t outputDim = output->getWidth(); CHECK_EQ(insNum, label->getSize()); if (supportWeight) { CHECK_EQ(insNum, weight->getHeight()); CHECK_EQ((size_t)1, weight->getWidth()); } CHECK(colIdx_ + (int32_t)outputDim >= 0 && colIdx_ - (int32_t)outputDim < 0) << "column index [" << colIdx_ << "] out of range [-" << outputDim << ", " << outputDim << ")"; realColumnIdx_ = 0; if (colIdx_ >= 0) { realColumnIdx_ = colIdx_; } else { realColumnIdx_ = outputDim + colIdx_; } if (dynamic_cast(output.get())) { Matrix::resizeOrCreate(cpuOutput_, insNum, outputDim, /* trans=*/false, /* useGpu=*/false); cpuOutput_->copyFrom(*output); IVector::resizeOrCreate(cpuLabel_, insNum, false); cpuLabel_->copyFrom(*label); if (supportWeight) { Matrix::resizeOrCreate(cpuWeight_, insNum, (size_t)1, false, false); cpuWeight_->copyFrom(*weight); } output = cpuOutput_; label = cpuLabel_; weight = cpuWeight_; } real* outputD = output->getData(); int* labelD = label->getData(); real* weightD = supportWeight ? weight->getData() : nullptr; size_t pos = realColumnIdx_; for (size_t i = 0; i < insNum; ++i) { real value = outputD[pos]; uint32_t binIdx = static_cast(value * kBinNum_); CHECK(binIdx <= kBinNum_) << "bin index [" << binIdx << "] out of range, predict value[" << value << "]"; real w = supportWeight ? weightD[i] : 1.0; if (labelD[i] == kNegativeLabel_) { statNeg_[binIdx] += w; } else { statPos_[binIdx] += w; } pos += outputDim; } return 0; } void AucEvaluator::distributeEval(ParameterClient2* client) { client->reduce(statPos_, statPos_, kBinNum_ + 1, FLAGS_trainer_id, 0); client->reduce(statNeg_, statNeg_, kBinNum_ + 1, FLAGS_trainer_id, 0); } double AucEvaluator::calcAuc() const { double totPos = 0.0; double totNeg = 0.0; double totPosPrev = 0.0; double totNegPrev = 0.0; double auc = 0.0; int64_t idx = kBinNum_; while (idx >= 0) { totPosPrev = totPos; totNegPrev = totNeg; totPos += statPos_[idx]; totNeg += statNeg_[idx]; auc += trapezoidArea(totNeg, totNegPrev, totPos, totPosPrev); --idx; } if (totPos > 0.0 && totNeg > 0.0) { return auc / totPos / totNeg; } else { return 0.0; } } real AucEvaluator::getValueImpl() const { return calcAuc(); } std::string AucEvaluator::getTypeImpl() const { if (colIdx_ == -1) { return "last-column-auc"; } else { return "auc"; } } // class RankAucEvaluator REGISTER_EVALUATOR(rankauc, RankAucEvaluator); void RankAucEvaluator::start() { Evaluator::start(); } void RankAucEvaluator::updateSamplesNum( const std::vector& arguments) { numSamples_ += arguments[0].getNumSequences(); } real RankAucEvaluator::evalImp(std::vector& arguments) { CHECK_GE(arguments.size(), 2U); CHECK_LE(arguments.size(), 3U); double batchAuc = 0.0; output_ = arguments[0].value; click_ = arguments[1].value; size_t batchSize = output_->getHeight(); CHECK(!output_->useGpu()) << "RankAUC evaluator does not support GPU!"; if (arguments.size() == 3U) { pv_ = arguments[2].value; } else { Matrix::resizeOrCreate(pv_, batchSize, 1, false, false); std::fill(pv_->getData(), pv_->getData() + batchSize, 1.0); } real* outputData = output_->getData(); real* clickData = click_->getData(); real* pvData = pv_->getData(); auto startPos = arguments[0].sequenceStartPositions->getVector(false); const int* startPosData = startPos->getData(); size_t batchNum = startPos->getSize() - 1; for (size_t i = 0; i < batchNum; ++i) { int beginPos = startPosData[i]; int endPos = startPosData[i + 1]; batchAuc += calcRankAuc(outputData + beginPos, clickData + beginPos, pvData + beginPos, endPos - beginPos); } return batchAuc; } double RankAucEvaluator::calcRankAuc(real* outputData, real* clickData, real* pvData, size_t size) { outputPair_.clear(); for (size_t i = 0; i < size; ++i) { outputPair_.push_back(std::make_pair(outputData[i], i)); } std::sort(outputPair_.begin(), outputPair_.end(), [](const std::pair& a, const std::pair& b) { return a.first > b.first; }); double aucTmp = 0.0; double clickSum = 0.0; double oldClickSum = 0.0; double noClick = 0.0; double noClickSum = 0.0; double lastScore = outputPair_[0].first + 1.0; for (size_t i = 0; i < size; ++i) { if (lastScore != outputPair_[i].first) { aucTmp += (clickSum + oldClickSum) * noClick / 2.0; oldClickSum = clickSum; noClick = 0.0; lastScore = outputPair_[i].first; } size_t id = outputPair_[i].second; noClick += pvData[id] - clickData[id]; noClickSum += noClick; clickSum += clickData[id]; } aucTmp += (clickSum + oldClickSum) * noClick / 2.0; return (clickSum * noClickSum) == 0.0 ? 0.0 : aucTmp / (clickSum * noClickSum); } std::string RankAucEvaluator::getTypeImpl() const { return "rankauc"; } // class PrecisionRecallEvaluator REGISTER_EVALUATOR(precision_recall, PrecisionRecallEvaluator); void PrecisionRecallEvaluator::start() { Evaluator::start(); statsInfo_.clear(); values_.clear(); } real PrecisionRecallEvaluator::evalImp(std::vector& arguments) { REGISTER_TIMER("PrecisionRecallEvaluator"); CHECK_GE(arguments.size(), (size_t)2); CHECK_LE(arguments.size(), (size_t)3); MatrixPtr output = arguments[0].value; IVectorPtr label = arguments[1].ids; MatrixPtr multiBinaryLabel = arguments[1].value; bool supportWeight = (3 == arguments.size()) ? true : false; MatrixPtr weight = supportWeight ? arguments[2].value : nullptr; if (nullptr == output || (nullptr == label && nullptr == multiBinaryLabel) || (supportWeight && nullptr == weight)) { return 0; } size_t insNum = output->getHeight(); size_t outputDim = output->getWidth(); if (label != nullptr) { CHECK_EQ(insNum, label->getSize()); } else { CHECK_EQ(insNum, multiBinaryLabel->getHeight()); CHECK_EQ(outputDim, multiBinaryLabel->getWidth()); } if (supportWeight) { CHECK_EQ(insNum, weight->getHeight()); CHECK_EQ((size_t)1, weight->getWidth()); } if (statsInfo_.size() != outputDim) { statsInfo_.clear(); statsInfo_.resize(outputDim); } isMultiBinaryLabel_ = (nullptr == label) ? true : false; if (label != nullptr) { if (dynamic_cast(output.get())) { Matrix::resizeOrCreate(cpuOutput_, insNum, outputDim, false, false); cpuOutput_->copyFrom(*output); IVector::resizeOrCreate(cpuLabel_, insNum, false); cpuLabel_->copyFrom(*label); if (supportWeight) { Matrix::resizeOrCreate(cpuWeight_, insNum, (size_t)1, false, false); cpuWeight_->copyFrom(*weight); } output = cpuOutput_; label = cpuLabel_; weight = cpuWeight_; } calcStatsInfo(output, label, weight); } else { // Not support GPU for multi binary labels CHECK(dynamic_cast(multiBinaryLabel.get())); calcStatsInfoMulti(output, multiBinaryLabel, weight); } return 0; } template void PrecisionRecallEvaluator::printStatsHelper(T1 labelCallback, T2 microAvgCallback) const { int label = config_.positive_label(); if (label != -1) { CHECK(label >= 0 && label < (int)statsInfo_.size()) << "positive_label [" << label << "] should be in range [0, " << statsInfo_.size() << ")"; double precision = calcPrecision(statsInfo_[label].TP, statsInfo_[label].FP); double recall = calcRecall(statsInfo_[label].TP, statsInfo_[label].FN); labelCallback(label, precision, recall, calcF1Score(precision, recall)); return; } // micro average method: precision = (TP1+TP2)/(TP1+FP1+TP2+FP2) // macro average method: precision = (precision1+precision2)/2 double microTotalTP = 0; double microTotalFP = 0; double microTotalFN = 0; double macroAvgPrecision = 0; double macroAvgRecall = 0; size_t numLabels = statsInfo_.size(); for (size_t i = 0; i < numLabels; ++i) { microTotalTP += statsInfo_[i].TP; microTotalFP += statsInfo_[i].FP; microTotalFN += statsInfo_[i].FN; macroAvgPrecision += calcPrecision(statsInfo_[i].TP, statsInfo_[i].FP); macroAvgRecall += calcRecall(statsInfo_[i].TP, statsInfo_[i].FN); } macroAvgPrecision /= numLabels; macroAvgRecall /= numLabels; double macroAvgF1Score = calcF1Score(macroAvgPrecision, macroAvgRecall); double microAvgPrecision = calcPrecision(microTotalTP, microTotalFP); double microAvgRecall = calcPrecision(microTotalTP, microTotalFN); double microAvgF1Score = calcF1Score(microAvgPrecision, microAvgRecall); microAvgCallback(macroAvgPrecision, macroAvgRecall, macroAvgF1Score, isMultiBinaryLabel_, microAvgPrecision, microAvgRecall, microAvgF1Score); } void PrecisionRecallEvaluator::printStats(std::ostream& os) const { this->printStatsHelper( [&os](int label, double precision, double recall, double f1) { os << "positive_label=" << label << " precision=" << precision << " recall=" << recall << " F1-score=" << f1; }, [&os](double macroAvgPrecision, double macroAvgRecall, double macroAvgF1Score, bool isMultiBinaryLabel, double microAvgPrecision, double microAvgRecall, double microAvgF1Score) { os << "macro-average-precision=" << macroAvgPrecision << " macro-average-recall=" << macroAvgRecall << " macro-average-F1-score=" << macroAvgF1Score; if (!isMultiBinaryLabel) { // precision and recall are equal in this case os << " micro-average-precision=" << microAvgPrecision; } else { os << " micro-average-precision=" << microAvgPrecision << " micro-average-recall=" << microAvgRecall << " micro-average-F1-score=" << microAvgF1Score; } }); } void PrecisionRecallEvaluator::calcStatsInfo(const MatrixPtr& output, const IVectorPtr& label, const MatrixPtr& weight) { size_t insNum = output->getHeight(); size_t dim = output->getWidth(); real* outputD = output->getData(); int* labelD = label->getData(); real* weightD = (weight != nullptr) ? weight->getData() : nullptr; for (size_t i = 0; i < insNum; ++i) { CHECK_GE(labelD[i], 0); CHECK_LT((size_t)labelD[i], dim); size_t maxIdx = 0; real maxValue = outputD[i * dim]; for (size_t j = 1; j < dim; ++j) { size_t idx = i * dim + j; if (maxValue < outputD[idx]) { maxIdx = j; maxValue = outputD[idx]; } } real w = (weightD != nullptr) ? weightD[i] : 1.0; if (maxIdx == (size_t)labelD[i]) { statsInfo_[maxIdx].TP += w; // true positive for labelD[i] // true negative for all labels except for labelD[i] for (size_t j = 0; j < dim; ++j) { statsInfo_[j].TN += w; } statsInfo_[maxIdx].TN -= w; } else { statsInfo_[labelD[i]].FN += w; // false negative for labelD[i] statsInfo_[maxIdx].FP += w; // false positive for maxIdx // true negatives for all labels except for maxIdx and labelD[i] for (size_t j = 0; j < dim; ++j) { statsInfo_[j].TN += w; } statsInfo_[maxIdx].TN -= w; statsInfo_[labelD[i]].TN -= w; } } } void PrecisionRecallEvaluator::calcStatsInfoMulti(const MatrixPtr& output, const MatrixPtr& label, const MatrixPtr& weight) { size_t insNum = output->getHeight(); size_t dim = output->getWidth(); real* outputD = output->getData(); auto labelD = dynamic_cast(label.get()); real* weightD = (weight != nullptr) ? weight->getData() : nullptr; real threshold = config_.classification_threshold(); for (size_t i = 0; i < insNum; ++i) { for (size_t j = 0; j < dim; ++j) { real w = (weightD != nullptr) ? weightD[i] : 1.0; size_t idx = i * dim + j; if (outputD[idx] < threshold) { statsInfo_[j].TN += w; // true negative } else { statsInfo_[j].FP += w; // false positive } } const int* cols = labelD->getRowCols(i); for (size_t j = 0; j < labelD->getColNum(i); ++j) { CHECK_LT(size_t(cols[j]), dim); real w = (weightD != nullptr) ? weightD[i] : 1.0; size_t idx = i * dim + cols[j]; if (outputD[idx] < threshold) { statsInfo_[cols[j]].FN += w; // false negative statsInfo_[cols[j]].TN -= w; // true negative } else { statsInfo_[cols[j]].TP += w; // true positive statsInfo_[cols[j]].FP -= w; // false positive } } } } void PrecisionRecallEvaluator::storeLocalValues() const { if (this->values_.size() == 0) { this->printStatsHelper( [this](int label, double precision, double recall, double f1) { values_["positive_label"] = (double)label; values_["precision"] = precision; values_["recal"] = recall; values_["F1-score"] = f1; }, [this](double macroAvgPrecision, double macroAvgRecall, double macroAvgF1Score, bool isMultiBinaryLabel, double microAvgPrecision, double microAvgRecall, double microAvgF1Score) { values_["macro-average-precision"] = macroAvgPrecision; values_["macro-average-recall"] = macroAvgRecall; values_["macro-average-F1-score"] = macroAvgF1Score; if (!isMultiBinaryLabel) { // precision and recall are equal in this case values_["micro-average-precision"] = microAvgPrecision; } else { values_["micro-average-precision"] = microAvgPrecision; values_["micro-average-recall"] = microAvgRecall; values_["micro-average-F1-score"] = microAvgF1Score; } }); } } void PrecisionRecallEvaluator::getNames(std::vector* names) { this->storeLocalValues(); names->clear(); names->reserve(this->values_.size()); for (auto it = this->values_.begin(); it != this->values_.end(); ++it) { names->push_back(this->config_.name() + "." + it->first); } } real PrecisionRecallEvaluator::getValue(const std::string& name, Error* err) const { this->storeLocalValues(); std::vector buffers; paddle::str::split(name, '.', &buffers); auto it = this->values_.find(buffers[buffers.size() - 1]); if (it != this->values_.end() && err != nullptr) { *err = Error("No such key %s", name.c_str()); return .0f; } return it->second; } std::string PrecisionRecallEvaluator::getType(const std::string& name, Error* err) const { Error localErr; if (err == nullptr) { err = &localErr; } this->getValue(name, err); if (!err->isOK()) { return ""; } return "precision_recall"; } void PrecisionRecallEvaluator::distributeEval(ParameterClient2* client) { size_t size = 4 * statsInfo_.size(); double* buf = new double[size]; for (size_t i = 0; i < statsInfo_.size(); ++i) { buf[4 * i + 0] = statsInfo_[i].TP; buf[4 * i + 1] = statsInfo_[i].TN; buf[4 * i + 2] = statsInfo_[i].FP; buf[4 * i + 3] = statsInfo_[i].FN; } client->reduce(buf, buf, size, FLAGS_trainer_id, 0); for (size_t i = 0; i < statsInfo_.size(); ++i) { statsInfo_[i].TP = buf[4 * i + 0]; statsInfo_[i].TN = buf[4 * i + 1]; statsInfo_[i].FP = buf[4 * i + 2]; statsInfo_[i].FN = buf[4 * i + 3]; } delete[] buf; } REGISTER_EVALUATOR(pnpair, PnpairEvaluator); void PnpairEvaluator::start() { Evaluator::start(); memset(pairArray_, 0, sizeof(pairArray_)); predictArray_.clear(); } real PnpairEvaluator::evalImp(std::vector& arguments) { CHECK_GE(arguments.size(), 3UL); CHECK_LE(arguments.size(), 4UL); MatrixPtr output = arguments[0].value; IVectorPtr label = arguments[1].ids; IVectorPtr info = arguments[2].ids; bool supportWeight = (4 == arguments.size()) ? true : false; MatrixPtr weight = supportWeight ? arguments[3].value : nullptr; if (nullptr == output || nullptr == label || (supportWeight && nullptr == weight)) { return 0; } size_t height = output->getHeight(); size_t width = output->getWidth(); CHECK_EQ(height, label->getSize()); CHECK_EQ(height, info->getSize()); if (supportWeight) { CHECK_EQ(height, weight->getHeight()); CHECK_EQ((size_t)1, weight->getWidth()); } if (dynamic_cast(output.get())) { Matrix::resizeOrCreate(cpuOutput_, height, width, false, false); IVector::resizeOrCreate(cpuLabel_, height, false); IVector::resizeOrCreate(cpuInfo_, height, false); cpuOutput_->copyFrom(*output); cpuLabel_->copyFrom(*label); cpuInfo_->copyFrom(*info); output = cpuOutput_; label = cpuLabel_; info = cpuInfo_; if (supportWeight) { Matrix::resizeOrCreate(cpuWeight_, height, (size_t)1, false, false); cpuWeight_->copyFrom(*weight); weight = cpuWeight_; } } real* outputs = output->getData(); int* labels = label->getData(); int* infos = info->getData(); real* weights = supportWeight ? weight->getData() : nullptr; for (size_t i = 0; i < output->getHeight(); i++) { real y1 = outputs[i * width + (width - 1)]; real w = supportWeight ? weights[i] : 1.0; predictArray_.push_back(PredictionResult(y1, labels[i], infos[i], w)); } return 0; } void PnpairEvaluator::stat(size_t start, size_t end, PredictionResult* answers, double& pos, double& neg, double& spe) { for (size_t i = start; i < end; i++) { for (size_t j = i + 1; j < end; j++) { CHECK_EQ(answers[i].queryid, answers[j].queryid); // The pair weight is the mean of the two samples' weight double weight = (answers[i].weight + answers[j].weight) / 2.0; if (answers[i].label != answers[j].label) { if ((answers[i].out > answers[j].out && answers[i].label > answers[j].label) || (answers[i].out < answers[j].out && answers[i].label < answers[j].label)) { pos += weight; } else if ((answers[i].out > answers[j].out && answers[i].label < answers[j].label) || (answers[i].out < answers[j].out && answers[i].label > answers[j].label)) { neg += weight; } else { spe += weight; } } } } } void PnpairEvaluator::calc(std::vector& predictArray) { std::sort(predictArray.begin(), predictArray.end(), [](const PredictionResult& x, const PredictionResult& y) { return x.queryid < y.queryid; }); double pos = 0; double neg = 0; double special = 0; auto start = predictArray.begin(); while (start != predictArray.end()) { auto end = std::find_if( start + 1, predictArray.end(), [=](const PredictionResult& x) { return x.queryid != start->queryid; }); CHECK(end != start); stat(start - predictArray.begin(), end - predictArray.begin(), predictArray.data(), pos, neg, special); start = end; } pairArray_[0] += pos; pairArray_[1] += neg; LOG(INFO) << " calc total pos pair: " << pos << " calc total neg pair: " << neg << " calc total special pair: " << special; } std::string PnpairEvaluator::getTypeImpl() const { return "pnpair"; } ClassRegistrar Evaluator::registrar_; Evaluator* Evaluator::create(const EvaluatorConfig& config) { Evaluator* evaluator = registrar_.createByType(config.type()); evaluator->init(config); return evaluator; } REGISTER_EVALUATOR(classification_error, ClassificationErrorEvaluator); REGISTER_EVALUATOR(sum, SumEvaluator); static InitFunction __reg_type_auc_sum__([]() { Evaluator::registrar_.registerClass( "last-column-sum", [] { return new ColumnSumEvaluator(-1); }); Evaluator::registrar_.registerClass("last-column-auc", [] { return new AucEvaluator(-1); }); }); /** * @brief print value of each layer. * * The config file api is value_printer_evaluator. */ class ValuePrinter : public Evaluator { public: virtual void eval(const NeuralNetwork& nn) { layerOutputs_.clear(); for (const std::string& name : config_.input_layers()) { auto& argu = nn.getLayer(name)->getOutput(); layerOutputs_[name] = std::unordered_map(); auto& out = layerOutputs_[name]; argu.getValueString(&out); for (auto field : {"value", "id", "sequence pos", "sub-sequence pos"}) { auto it = out.find(field); if (it != out.end()) { LOG(INFO) << "layer=" << name << " " << field << ":\n" << it->second; } } } } virtual void updateSamplesNum(const std::vector& arguments) {} virtual real evalImp(std::vector& arguments) { return 0; } private: std::unordered_map> layerOutputs_; // Evaluator interface public: void getNames(std::vector* names) { for (auto layerIt = layerOutputs_.begin(); layerIt != layerOutputs_.end(); ++layerIt) { for (auto it = layerIt->second.begin(); it != layerIt->second.end(); ++it) { names->push_back(config_.name() + "." + layerIt->first + "." + it->second); } } } real getValue(const std::string& name, Error* err) const { (void)(name); if (err != nullptr) { *err = Error( "ValuePrinter do not support getValue, use getValueString instead."); } return .0f; } std::string getValueStr(const std::string& name, Error* err) const { std::vector buffer; str::split(name, '.', &buffer); if (buffer.size() < 2) { if (err != nullptr) { *err = Error("No such key %s", name.c_str()); } return ""; } auto fieldName = buffer[buffer.size() - 1]; auto layerName = buffer[buffer.size() - 2]; auto layerIt = layerOutputs_.find(layerName); if (layerIt == layerOutputs_.end()) { if (err != nullptr) { *err = Error("No such layer %s", layerName.c_str()); } return ""; } auto fieldIt = layerIt->second.find(fieldName); if (fieldIt == layerIt->second.end()) { if (err != nullptr) { *err = Error("No such value field %s", fieldName.c_str()); } return ""; } return fieldIt->second; } std::string getType(const std::string& name, Error* err) const { Error localErr; if (err == nullptr) { err = &localErr; } this->getValueStr(name, err); if (!err->isOK()) { return ""; } return "value_printer"; } }; REGISTER_EVALUATOR(value_printer, ValuePrinter); /** * @brief print gradient of each layer. * * The config file api is gradient_printer_evaluator. */ class GradientPrinter : public Evaluator { public: GradientPrinter() {} virtual void eval(const NeuralNetwork& nn) { for (const std::string& name : config_.input_layers()) { const Argument& argu = nn.getLayer(name)->getOutput(); if (argu.grad) { std::ostringstream os; argu.grad->print(os); LOG(INFO) << "layer=" << name << " grad matrix:\n" << os.str(); } if (auto startPos = argu.sequenceStartPositions) { std::ostringstream os; startPos->getVector(false)->print(os, startPos->getSize()); LOG(INFO) << "layer=" << name << " sequence pos vector:\n" << os.str(); } } } virtual void updateSamplesNum(const std::vector& arguments) {} virtual real evalImp(std::vector& arguments) { return 0; } }; REGISTER_EVALUATOR(gradient_printer, GradientPrinter); /** * @brief print row max id vctor of each layer * * The config file api is maxid_printer_evaluator. */ class MaxIdPrinter : public Evaluator { private: IVectorPtr maxIds_; MatrixPtr maxValues_; public: MaxIdPrinter() {} virtual void eval(const NeuralNetwork& nn) { for (const std::string& name : config_.input_layers()) { const Argument& argu = nn.getLayer(name)->getOutput(); if (argu.value) { size_t height = argu.value->getHeight(); size_t width = config_.num_results(); IVector::resizeOrCreate(maxIds_, height * width, false); Matrix::resizeOrCreate(maxValues_, height, width, false); argu.value->rowMax(*maxIds_, *maxValues_); std::ostringstream os; int* ids = maxIds_->getData(); real* values = maxValues_->getData(); for (size_t i = 0; i < height; ++i) { for (size_t j = 0; j < width; ++j) { size_t pos = i * width + j; os << ids[pos] << " : " << values[pos] << ", "; } os << std::endl; } LOG(INFO) << "layer=" << name << " row max id vector:\n" << os.str(); } } } virtual void updateSamplesNum(const std::vector& arguments) {} virtual real evalImp(std::vector& arguments) { return 0; } }; REGISTER_EVALUATOR(max_id_printer, MaxIdPrinter); /** * @brief print sequence max frames of each layer * * The config file api is maxframe_printer_evaluator. */ class MaxFramePrinter : public Evaluator { private: IVectorPtr maxIds_; MatrixPtr maxValues_; MatrixPtr value_; public: MaxFramePrinter() { value_ = Matrix::create(nullptr, /* height= */ 1, 1, /* trans= */ false, false); } virtual void eval(const NeuralNetwork& nn) { for (const std::string& name : config_.input_layers()) { const Argument& argu = nn.getLayer(name)->getOutput(); CHECK_EQ(argu.value->getWidth(), 1LU); size_t numSequences = argu.getNumSequences(); const int* starts = argu.sequenceStartPositions->getData(false); std::ostringstream os; for (size_t i = 0; i < numSequences; ++i) { size_t offset = starts[i]; size_t size = starts[i + 1] - starts[i]; value_->setData(argu.value->getData() + offset, 1LU, size); size_t height = 1LU; size_t width = std::min((size_t)config_.num_results(), size); IVector::resizeOrCreate(maxIds_, height * width, false); Matrix::resizeOrCreate(maxValues_, height, width, false); value_->rowMax(*maxIds_, *maxValues_); int* ids = maxIds_->getData(); real* values = maxValues_->getData(); for (size_t j = 0; j < width; ++j) { os << ids[j] << " : " << values[j] << ", "; } os << "total " << size << " frames" << std::endl; } LOG(INFO) << "layer=" << name << " sequence max frames:\n" << os.str(); } } virtual void updateSamplesNum(const std::vector& arguments) {} virtual real evalImp(std::vector& arguments) { return 0; } }; REGISTER_EVALUATOR(max_frame_printer, MaxFramePrinter); /** * @brief print text according to index matrix and a dictionary. * * There can be multiple input to this layer: * - If there is only one input, the input must be a matrix containing * the sequence of indices; * - If there are more than one input, the first input should be ids, * and are interpreted as sample ids. * * The output format will be: * * - sequence without sub-sequence, and there is probability. * * @code * id \t prob space_seperated_tokens_from_dictionary_according_to_seq * @endcode * * - sequence without sub-sequence, and there is not probability. * * @code * id \t space_seperated_tokens_from_dictionary_according_to_seq * @endcode * * - sequence with sub-sequence, and there is not probability. * * @code * id \t space_seperated_tokens_from_dictionary_according_to_sub_seq * \t \t space_seperated_tokens_from_dictionary_according_to_sub_seq * ... * @endcode * * Typically SequenceTextPrinter layer takes output of maxid or RecurrentGroup * with maxid (when generating) as an input. * * The config file api is seqtext_printer_evaluator. * */ class SequenceTextPrinter : public Evaluator { private: /// dict_file, which contains a list of tokens std::vector dict_; /// result_file, which is the output file std::ofstream os_; /// True/False, to indicate whether to use space to separate output tokens. /// Default is True. No space is added if set to False. bool delimited_; /// store the cpu version of argument.ids std::vector cpuIds_; /// store the probability associated with each sequence std::vector cpuIn_; public: SequenceTextPrinter() {} virtual void init(const EvaluatorConfig& config) { Evaluator::init(config); if (!config.dict_file().empty()) { loadFileList(config.dict_file(), dict_); } os_.open(config.result_file(), std::ofstream::trunc); CHECK(os_.is_open()) << "Failed to open file " << config.result_file(); delimited_ = config.delimited(); } virtual void updateSamplesNum(const std::vector& arguments) {} virtual real evalImp(std::vector& arguments) { CHECK_GE(arguments.size(), 1LU); bool hasId = arguments.size() > 1; size_t numSequences = arguments[0].getNumSequences(); if (hasId) { CHECK_EQ(arguments[0].ids->getSize(), numSequences) << "first input must be sample id."; } for (size_t i = hasId ? 1 : 0; i < arguments.size(); ++i) { CHECK_EQ((size_t)arguments[i].getNumSequences(), numSequences); } auto resizeVector = [](IVectorPtr& dest, const IVectorPtr& src) { if (src && src->useGpu()) { IVector::resizeOrCreate(dest, src->getSize(), false); dest->copyFrom(*src); } else { dest = src; } }; auto resizeMatrix = [](MatrixPtr& dest, const MatrixPtr& src) { if (src && src->useGpu()) { Matrix::resizeOrCreate( dest, src->getHeight(), src->getWidth(), false, false); dest->copyFrom(*src); } else { dest = src; } }; cpuIds_.resize(arguments.size()); cpuIn_.resize(arguments.size()); for (size_t i = 0; i < arguments.size(); ++i) { resizeVector(cpuIds_[i], arguments[i].ids); resizeMatrix(cpuIn_[i], arguments[i].in); } int* sampleIds = nullptr; if (hasId) { sampleIds = cpuIds_[0]->getData(); } for (size_t i = 0; i < numSequences; ++i) { os_ << (hasId ? sampleIds[i] : i); for (size_t j = hasId ? 1 : 0; j < arguments.size(); ++j) { int* output = cpuIds_[j]->getData(); const int* starts = arguments[j].sequenceStartPositions->getData(false); auto seqPrint = [&](int start, int end) { os_ << "\t"; for (int k = start; k < end; k++) { int id = output[k]; os_ << (delimited_ ? " " : ""); if (!dict_.empty()) { CHECK_LT((size_t)id, dict_.size()); os_ << dict_[id]; } else { os_ << id; } } }; if (arguments[j].hasSubseq()) { // print sequence with sub-sequence const int* subStarts = arguments[j].subSequenceStartPositions->getData(false); int subSeqId_start = 0; int subSeqId_end = 0; for (size_t k = 0; k < (size_t)arguments[j].getNumSubSequences() + 1; ++k) { if (starts[i] == subStarts[k]) subSeqId_start = k; if (starts[i + 1] == subStarts[k]) subSeqId_end = k; } for (int k = subSeqId_start; k < subSeqId_end; k++) { seqPrint(subStarts[k], subStarts[k + 1]); os_ << std::endl; } } else { // print sequence without sub-sequence if (arguments[j].in) { // beam print real* probs = cpuIn_[j]->rowBuf(i); os_ << std::endl; int start = starts[i]; int seqEnd = starts[i + 1]; for (size_t k = 0; k < arguments[j].in->getWidth(); ++k) { if (start == seqEnd) { break; } int end = start + output[start] + 2; CHECK_LE(end, seqEnd); CHECK_EQ(output[end - 1], -1); os_ << k << "\t" << probs[k]; seqPrint(start + 1, end - 1); os_ << std::endl; start = end; } } else { seqPrint(starts[i], starts[i + 1]); } } } os_ << std::endl; } return 0; } }; REGISTER_EVALUATOR(seq_text_printer, SequenceTextPrinter); /** * @brief print classification error. * * The config file api is classification_error_printer_evaluator. */ class ClassificationErrorPrinter : public ClassificationErrorEvaluator { public: virtual void updateSamplesNum(const std::vector& arguments) {} virtual real evalImp(std::vector& arguments) { MatrixPtr errorMat = calcError(arguments); std::ostringstream os; errorMat->print(os); LOG(INFO) << "Printer=" << config_.name() << " Classification Error:\n" << os.str(); if (auto startPos = arguments[0].sequenceStartPositions) { std::ostringstream os; startPos->getVector(false)->print(os, startPos->getSize()); LOG(INFO) << "Printer=" << config_.name() << " sequence pos vector:\n" << os.str(); } return 0; } }; REGISTER_EVALUATOR(classification_error_printer, ClassificationErrorPrinter); std::string DummyEvaluator::getTypeImpl() const { return "dummy"; } } // namespace paddle