/*
* Copyright (C) 2018 The Android Open Source Project
*
* 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 "lang_id/common/fel/feature-extractor.h"
#include "lang_id/common/fel/feature-types.h"
#include "lang_id/common/fel/fel-parser.h"
#include "lang_id/common/lite_base/logging.h"
#include "lang_id/common/lite_strings/numbers.h"
namespace libtextclassifier3 {
namespace mobile {
constexpr FeatureValue GenericFeatureFunction::kNone;
GenericFeatureExtractor::GenericFeatureExtractor() {}
GenericFeatureExtractor::~GenericFeatureExtractor() {}
bool GenericFeatureExtractor::Parse(const string &source) {
// Parse feature specification into descriptor.
FELParser parser;
if (!parser.Parse(source, mutable_descriptor())) {
SAFTM_LOG(ERROR) << "Error parsing the FEL spec " << source;
return false;
}
// Initialize feature extractor from descriptor.
return InitializeFeatureFunctions();
}
bool GenericFeatureExtractor::InitializeFeatureTypes() {
// Register all feature types.
GetFeatureTypes(&feature_types_);
for (size_t i = 0; i < feature_types_.size(); ++i) {
FeatureType *ft = feature_types_[i];
ft->set_base(i);
// Check for feature space overflow.
double domain_size = ft->GetDomainSize();
if (domain_size < 0) {
SAFTM_LOG(ERROR) << "Illegal domain size for feature " << ft->name()
<< ": " << domain_size;
return false;
}
}
return true;
}
string GenericFeatureFunction::GetParameter(const string &name,
const string &default_value) const {
// Find named parameter in feature descriptor.
for (int i = 0; i < descriptor_->parameter_size(); ++i) {
if (name == descriptor_->parameter(i).name()) {
return descriptor_->parameter(i).value();
}
}
return default_value;
}
GenericFeatureFunction::GenericFeatureFunction() {}
GenericFeatureFunction::~GenericFeatureFunction() { delete feature_type_; }
int GenericFeatureFunction::GetIntParameter(const string &name,
int default_value) const {
string value_str = GetParameter(name, "");
if (value_str.empty()) {
// Parameter not specified, use default value for it.
return default_value;
}
int value = 0;
if (!LiteAtoi(value_str, &value)) {
SAFTM_LOG(DFATAL) << "Unable to parse '" << value_str
<< "' as int for parameter " << name;
return default_value;
}
return value;
}
bool GenericFeatureFunction::GetBoolParameter(const string &name,
bool default_value) const {
string value = GetParameter(name, "");
if (value.empty()) return default_value;
if (value == "true") return true;
if (value == "false") return false;
SAFTM_LOG(DFATAL) << "Illegal value '" << value << "' for bool parameter "
<< name;
return default_value;
}
void GenericFeatureFunction::GetFeatureTypes(
std::vector<FeatureType *> *types) const {
if (feature_type_ != nullptr) types->push_back(feature_type_);
}
FeatureType *GenericFeatureFunction::GetFeatureType() const {
// If a single feature type has been registered return it.
if (feature_type_ != nullptr) return feature_type_;
// Get feature types for function.
std::vector<FeatureType *> types;
GetFeatureTypes(&types);
// If there is exactly one feature type return this, else return null.
if (types.size() == 1) return types[0];
return nullptr;
}
string GenericFeatureFunction::name() const {
string output;
if (descriptor_->name().empty()) {
if (!prefix_.empty()) {
output.append(prefix_);
output.append(".");
}
ToFEL(*descriptor_, &output);
} else {
output = descriptor_->name();
}
return output;
}
} // namespace mobile
} // namespace nlp_saft