/* * Copyright (C) 2015, 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 "fake_io_delegate.h" #include <android-base/stringprintf.h> #include <android-base/strings.h> #include "logging.h" #include "os.h" #include "tests/test_util.h" using android::base::StringAppendF; using android::base::StringPrintf; using std::string; using std::unique_ptr; using std::vector; namespace android { namespace aidl { namespace test { // Claims to always write successfully, but can't close the file. class BrokenCodeWriter : public CodeWriter { bool Write(const char* /* format */, ...) override { return true; } bool Close() override { return false; } ~BrokenCodeWriter() override = default; }; // class BrokenCodeWriter unique_ptr<string> FakeIoDelegate::GetFileContents( const string& relative_filename, const string& content_suffix) const { string filename = CleanPath(relative_filename); unique_ptr<string> contents; auto it = file_contents_.find(filename); if (it == file_contents_.end()) { return contents; } contents.reset(new string); *contents = it->second; contents->append(content_suffix); return contents; } unique_ptr<LineReader> FakeIoDelegate::GetLineReader( const string& file_path) const { unique_ptr<LineReader> ret; const auto& it = file_contents_.find(CleanPath(file_path)); if (it != file_contents_.cend()) { ret = LineReader::ReadFromMemory(it->second); } return ret; } bool FakeIoDelegate::FileIsReadable(const string& path) const { return file_contents_.find(CleanPath(path)) != file_contents_.end(); } std::unique_ptr<CodeWriter> FakeIoDelegate::GetCodeWriter( const std::string& file_path) const { if (broken_files_.count(file_path) > 0) { return unique_ptr<CodeWriter>(new BrokenCodeWriter); } removed_files_.erase(file_path); written_file_contents_[file_path] = ""; return CodeWriter::ForString(&written_file_contents_[file_path]); } void FakeIoDelegate::RemovePath(const std::string& file_path) const { removed_files_.insert(file_path); } void FakeIoDelegate::SetFileContents(const string& filename, const string& contents) { file_contents_[filename] = contents; } vector<string> FakeIoDelegate::ListFiles(const string& dir) const { const string dir_name = dir.back() == OS_PATH_SEPARATOR ? dir : dir + OS_PATH_SEPARATOR; vector<string> files; for (auto it = file_contents_.begin(); it != file_contents_.end(); it++) { if (android::base::StartsWith(it->first, dir_name)) { files.emplace_back(it->first); } } return files; } void FakeIoDelegate::AddStubParcelable(const string& canonical_name, const string& cpp_header) { string package, class_name, rel_path; SplitPackageClass(canonical_name, &rel_path, &package, &class_name); string contents; if (cpp_header.empty()) { contents = StringPrintf("package %s;\nparcelable %s;", package.c_str(), class_name.c_str()); } else { contents = StringPrintf("package %s;\nparcelable %s cpp_header \"%s\";", package.c_str(), class_name.c_str(), cpp_header.c_str()); } SetFileContents(rel_path, contents); } void FakeIoDelegate::AddStubInterface(const string& canonical_name) { string package, class_name, rel_path; SplitPackageClass(canonical_name, &rel_path, &package, &class_name); string contents = StringPrintf("package %s;\ninterface %s { }", package.c_str(), class_name.c_str()); SetFileContents(rel_path, contents); } void FakeIoDelegate::AddCompoundParcelable(const string& canonical_name, const vector<string>& subclasses) { string package, class_name, rel_path; SplitPackageClass(canonical_name, &rel_path, &package, &class_name); string contents = StringPrintf("package %s;\n", package.c_str()); for (const string& subclass : subclasses) { StringAppendF(&contents, "parcelable %s.%s;\n", class_name.c_str(), subclass.c_str()); } SetFileContents(rel_path, contents); } void FakeIoDelegate::AddBrokenFilePath(const std::string& path) { broken_files_.insert(path); } bool FakeIoDelegate::GetWrittenContents(const string& path, string* content) { const auto it = written_file_contents_.find(path); if (it == written_file_contents_.end()) { return false; } if (content) { *content = it->second; } return true; } std::vector<std::string> FakeIoDelegate::ListOutputFiles() { std::vector<std::string> out; for (const auto& [file, contents] : written_file_contents_) { out.push_back(file); } return out; } bool FakeIoDelegate::PathWasRemoved(const std::string& path) { if (removed_files_.count(path) > 0) { return true; } return false; } string FakeIoDelegate::CleanPath(const string& path) const { string clean_path = path; while (clean_path.length() >= 2 && clean_path[0] == '.' && clean_path[1] == OS_PATH_SEPARATOR) { clean_path = clean_path.substr(2); } return clean_path; } } // namespace test } // namespace aidl } // namespace android