/*
* 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 "actions/zlib-utils.h"
#include <memory>
#include "utils/base/logging.h"
#include "utils/intents/zlib-utils.h"
#include "utils/resources.h"
namespace libtextclassifier3 {
// Compress rule fields in the model.
bool CompressActionsModel(ActionsModelT* model) {
std::unique_ptr<ZlibCompressor> zlib_compressor = ZlibCompressor::Instance();
if (!zlib_compressor) {
TC3_LOG(ERROR) << "Cannot compress model.";
return false;
}
// Compress regex rules.
if (model->rules != nullptr) {
for (int i = 0; i < model->rules->rule.size(); i++) {
RulesModel_::RuleT* rule = model->rules->rule[i].get();
rule->compressed_pattern.reset(new CompressedBufferT);
zlib_compressor->Compress(rule->pattern, rule->compressed_pattern.get());
rule->pattern.clear();
}
}
if (model->low_confidence_rules != nullptr) {
for (int i = 0; i < model->low_confidence_rules->rule.size(); i++) {
RulesModel_::RuleT* rule = model->low_confidence_rules->rule[i].get();
if (!rule->pattern.empty()) {
rule->compressed_pattern.reset(new CompressedBufferT);
zlib_compressor->Compress(rule->pattern,
rule->compressed_pattern.get());
rule->pattern.clear();
}
if (!rule->output_pattern.empty()) {
rule->compressed_output_pattern.reset(new CompressedBufferT);
zlib_compressor->Compress(rule->pattern,
rule->compressed_output_pattern.get());
rule->output_pattern.clear();
}
}
}
if (!model->lua_actions_script.empty()) {
model->compressed_lua_actions_script.reset(new CompressedBufferT);
zlib_compressor->Compress(model->lua_actions_script,
model->compressed_lua_actions_script.get());
}
if (model->ranking_options != nullptr &&
!model->ranking_options->lua_ranking_script.empty()) {
model->ranking_options->compressed_lua_ranking_script.reset(
new CompressedBufferT);
zlib_compressor->Compress(
model->ranking_options->lua_ranking_script,
model->ranking_options->compressed_lua_ranking_script.get());
}
// Compress resources.
if (model->resources != nullptr) {
CompressResources(model->resources.get());
}
// Compress intent generator.
if (model->android_intent_options != nullptr) {
CompressIntentModel(model->android_intent_options.get());
}
return true;
}
bool DecompressActionsModel(ActionsModelT* model) {
std::unique_ptr<ZlibDecompressor> zlib_decompressor =
ZlibDecompressor::Instance();
if (!zlib_decompressor) {
TC3_LOG(ERROR) << "Cannot initialize decompressor.";
return false;
}
// Decompress regex rules.
if (model->rules != nullptr) {
for (int i = 0; i < model->rules->rule.size(); i++) {
RulesModel_::RuleT* rule = model->rules->rule[i].get();
if (!zlib_decompressor->MaybeDecompress(rule->compressed_pattern.get(),
&rule->pattern)) {
TC3_LOG(ERROR) << "Cannot decompress pattern: " << i;
return false;
}
rule->compressed_pattern.reset(nullptr);
}
}
// Decompress low confidence rules.
if (model->low_confidence_rules != nullptr) {
for (int i = 0; i < model->low_confidence_rules->rule.size(); i++) {
RulesModel_::RuleT* rule = model->low_confidence_rules->rule[i].get();
if (!zlib_decompressor->MaybeDecompress(rule->compressed_pattern.get(),
&rule->pattern)) {
TC3_LOG(ERROR) << "Cannot decompress pattern: " << i;
return false;
}
if (!zlib_decompressor->MaybeDecompress(
rule->compressed_output_pattern.get(), &rule->output_pattern)) {
TC3_LOG(ERROR) << "Cannot decompress pattern: " << i;
return false;
}
rule->compressed_pattern.reset(nullptr);
rule->compressed_output_pattern.reset(nullptr);
}
}
if (!zlib_decompressor->MaybeDecompress(
model->compressed_lua_actions_script.get(),
&model->lua_actions_script)) {
TC3_LOG(ERROR) << "Cannot decompress actions script.";
return false;
}
if (model->ranking_options != nullptr &&
!zlib_decompressor->MaybeDecompress(
model->ranking_options->compressed_lua_ranking_script.get(),
&model->ranking_options->lua_ranking_script)) {
TC3_LOG(ERROR) << "Cannot decompress actions script.";
return false;
}
return true;
}
std::string CompressSerializedActionsModel(const std::string& model) {
std::unique_ptr<ActionsModelT> unpacked_model =
UnPackActionsModel(model.c_str());
TC3_CHECK(unpacked_model != nullptr);
TC3_CHECK(CompressActionsModel(unpacked_model.get()));
flatbuffers::FlatBufferBuilder builder;
FinishActionsModelBuffer(builder,
ActionsModel::Pack(builder, unpacked_model.get()));
return std::string(reinterpret_cast<const char*>(builder.GetBufferPointer()),
builder.GetSize());
}
bool GetUncompressedString(const flatbuffers::String* uncompressed_buffer,
const CompressedBuffer* compressed_buffer,
ZlibDecompressor* decompressor, std::string* out) {
if (uncompressed_buffer == nullptr && compressed_buffer == nullptr) {
out->clear();
return true;
}
return decompressor->MaybeDecompressOptionallyCompressedBuffer(
uncompressed_buffer, compressed_buffer, out);
}
} // namespace libtextclassifier3