// // Copyright 2022 The Abseil Authors. // // 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 // // https://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. // // Tests for stripping of literal strings. // --------------------------------------- // // When a `LOG` statement can be trivially proved at compile time to never fire, // e.g. due to `ABSL_MIN_LOG_LEVEL`, `NDEBUG`, or some explicit condition, data // streamed in can be dropped from the compiled program completely if they are // not used elsewhere. This most commonly affects string literals, which users // often want to strip to reduce binary size and/or redact information about // their program's internals (e.g. in a release build). // // These tests log strings and then validate whether they appear in the compiled // binary. This is done by opening the file corresponding to the running test // and running a simple string search on its contents. The strings to be logged // and searched for must be unique, and we must take care not to emit them into // the binary in any other place, e.g. when searching for them. The latter is // accomplished by computing them using base64; the source string appears in the // binary but the target string is computed at runtime. #include #if defined(__MACH__) #include #elif defined(_WIN32) #include #include #endif #include #include #include #include #include #include "gmock/gmock.h" #include "gtest/gtest.h" #include "absl/base/internal/strerror.h" #include "absl/flags/internal/program_name.h" #include "absl/log/check.h" #include "absl/log/internal/test_helpers.h" #include "absl/log/log.h" #include "absl/strings/escaping.h" #include "absl/strings/str_format.h" #include "absl/strings/string_view.h" namespace { using ::testing::_; using ::testing::Eq; using ::testing::NotNull; using absl::log_internal::kAbslMinLogLevel; std::string Base64UnescapeOrDie(absl::string_view data) { std::string decoded; CHECK(absl::Base64Unescape(data, &decoded)); return decoded; } // ----------------------------------------------------------------------------- // A Googletest matcher which searches the running binary for a given string // ----------------------------------------------------------------------------- // This matcher is used to validate that literal strings streamed into // `LOG` statements that ought to be compiled out (e.g. `LOG_IF(INFO, false)`) // do not appear in the binary. // // Note that passing the string to be sought directly to `FileHasSubstr()` all // but forces its inclusion in the binary regardless of the logging library's // behavior. For example: // // LOG_IF(INFO, false) << "you're the man now dog"; // // This will always pass: // // EXPECT_THAT(fp, FileHasSubstr("you're the man now dog")); // // So use this instead: // EXPECT_THAT(fp, FileHasSubstr( // Base64UnescapeOrDie("eW91J3JlIHRoZSBtYW4gbm93IGRvZw=="))); class FileHasSubstrMatcher final : public ::testing::MatcherInterface { public: explicit FileHasSubstrMatcher(absl::string_view needle) : needle_(needle) {} bool MatchAndExplain( FILE* fp, ::testing::MatchResultListener* listener) const override { std::string buf( std::max(needle_.size() * 2, 163840000), '\0'); size_t buf_start_offset = 0; // The file offset of the byte at `buf[0]`. size_t buf_data_size = 0; // The number of bytes of `buf` which contain // data. ::fseek(fp, 0, SEEK_SET); while (true) { // Fill the buffer to capacity or EOF: while (buf_data_size < buf.size()) { const size_t ret = fread(&buf[buf_data_size], sizeof(char), buf.size() - buf_data_size, fp); if (ret == 0) break; buf_data_size += ret; } if (ferror(fp)) { *listener << "error reading file"; return false; } const absl::string_view haystack(&buf[0], buf_data_size); const auto off = haystack.find(needle_); if (off != haystack.npos) { *listener << "string found at offset " << buf_start_offset + off; return true; } if (feof(fp)) { *listener << "string not found"; return false; } // Copy the end of `buf` to the beginning so we catch matches that span // buffer boundaries. `buf` and `buf_data_size` are always large enough // that these ranges don't overlap. memcpy(&buf[0], &buf[buf_data_size - needle_.size()], needle_.size()); buf_start_offset += buf_data_size - needle_.size(); buf_data_size = needle_.size(); } } void DescribeTo(std::ostream* os) const override { *os << "contains the string \"" << needle_ << "\" (base64(\"" << Base64UnescapeOrDie(needle_) << "\"))"; } void DescribeNegationTo(std::ostream* os) const override { *os << "does not "; DescribeTo(os); } private: std::string needle_; }; class StrippingTest : public ::testing::Test { protected: void SetUp() override { #ifndef NDEBUG // Non-optimized builds don't necessarily eliminate dead code at all, so we // don't attempt to validate stripping against such builds. GTEST_SKIP() << "StrippingTests skipped since this build is not optimized"; #elif defined(__EMSCRIPTEN__) // These tests require a way to examine the running binary and look for // strings; there's no portable way to do that. GTEST_SKIP() << "StrippingTests skipped since this platform is not optimized"; #endif } // Opens this program's executable file. Returns `nullptr` and writes to // `stderr` on failure. std::unique_ptr> OpenTestExecutable() { #if defined(__linux__) std::unique_ptr> fp( fopen("/proc/self/exe", "rb"), [](FILE* fp) { fclose(fp); }); if (!fp) { const std::string err = absl::base_internal::StrError(errno); absl::FPrintF(stderr, "Failed to open /proc/self/exe: %s\n", err); } return fp; #elif defined(__Fuchsia__) // TODO(b/242579714): We need to restore the test coverage on this platform. std::unique_ptr> fp( fopen(absl::StrCat("/pkg/bin/", absl::flags_internal::ShortProgramInvocationName()) .c_str(), "rb"), [](FILE* fp) { fclose(fp); }); if (!fp) { const std::string err = absl::base_internal::StrError(errno); absl::FPrintF(stderr, "Failed to open /pkg/bin/: %s\n", err); } return fp; #elif defined(__MACH__) uint32_t size = 0; int ret = _NSGetExecutablePath(nullptr, &size); if (ret != -1) { absl::FPrintF(stderr, "Failed to get executable path: " "_NSGetExecutablePath(nullptr) returned %d\n", ret); return nullptr; } std::string path(size, '\0'); ret = _NSGetExecutablePath(&path[0], &size); if (ret != 0) { absl::FPrintF( stderr, "Failed to get executable path: _NSGetExecutablePath(buffer) " "returned %d\n", ret); return nullptr; } std::unique_ptr> fp( fopen(path.c_str(), "rb"), [](FILE* fp) { fclose(fp); }); if (!fp) { const std::string err = absl::base_internal::StrError(errno); absl::FPrintF(stderr, "Failed to open executable at %s: %s\n", path, err); } return fp; #elif defined(_WIN32) std::basic_string path(4096, _T('\0')); while (true) { const uint32_t ret = ::GetModuleFileName(nullptr, &path[0], path.size()); if (ret == 0) { absl::FPrintF( stderr, "Failed to get executable path: GetModuleFileName(buffer) " "returned 0\n"); return nullptr; } if (ret < path.size()) break; path.resize(path.size() * 2, _T('\0')); } std::unique_ptr> fp( _tfopen(path.c_str(), "rb"), [](FILE* fp) { fclose(fp); }); if (!fp) absl::FPrintF(stderr, "Failed to open executable\n"); return fp; #else absl::FPrintF(stderr, "OpenTestExecutable() unimplemented on this platform\n"); return nullptr; #endif } ::testing::Matcher FileHasSubstr(absl::string_view needle) { return MakeMatcher(new FileHasSubstrMatcher(needle)); } }; // This tests whether out methodology for testing stripping works on this // platform by looking for one string that definitely ought to be there and one // that definitely ought not to. If this fails, none of the `StrippingTest`s // are going to produce meaningful results. TEST_F(StrippingTest, Control) { constexpr char kEncodedPositiveControl[] = "U3RyaXBwaW5nVGVzdC5Qb3NpdGl2ZUNvbnRyb2w="; const std::string encoded_negative_control = absl::Base64Escape("StrippingTest.NegativeControl"); // Verify this mainly so we can encode other strings and know definitely they // won't encode to `kEncodedPositiveControl`. EXPECT_THAT(Base64UnescapeOrDie("U3RyaXBwaW5nVGVzdC5Qb3NpdGl2ZUNvbnRyb2w="), Eq("StrippingTest.PositiveControl")); auto exe = OpenTestExecutable(); ASSERT_THAT(exe, NotNull()); EXPECT_THAT(exe.get(), FileHasSubstr(kEncodedPositiveControl)); EXPECT_THAT(exe.get(), Not(FileHasSubstr(encoded_negative_control))); } TEST_F(StrippingTest, Literal) { // We need to load a copy of the needle string into memory (so we can search // for it) without leaving it lying around in plaintext in the executable file // as would happen if we used a literal. We might (or might not) leave it // lying around later; that's what the tests are for! const std::string needle = absl::Base64Escape("StrippingTest.Literal"); LOG(INFO) << "U3RyaXBwaW5nVGVzdC5MaXRlcmFs"; auto exe = OpenTestExecutable(); ASSERT_THAT(exe, NotNull()); if (absl::LogSeverity::kInfo >= kAbslMinLogLevel) { EXPECT_THAT(exe.get(), FileHasSubstr(needle)); } else { EXPECT_THAT(exe.get(), Not(FileHasSubstr(needle))); } } TEST_F(StrippingTest, LiteralInExpression) { // We need to load a copy of the needle string into memory (so we can search // for it) without leaving it lying around in plaintext in the executable file // as would happen if we used a literal. We might (or might not) leave it // lying around later; that's what the tests are for! const std::string needle = absl::Base64Escape("StrippingTest.LiteralInExpression"); LOG(INFO) << absl::StrCat("secret: ", "U3RyaXBwaW5nVGVzdC5MaXRlcmFsSW5FeHByZXNzaW9u"); std::unique_ptr> exe = OpenTestExecutable(); ASSERT_THAT(exe, NotNull()); if (absl::LogSeverity::kInfo >= kAbslMinLogLevel) { EXPECT_THAT(exe.get(), FileHasSubstr(needle)); } else { EXPECT_THAT(exe.get(), Not(FileHasSubstr(needle))); } } TEST_F(StrippingTest, Fatal) { // We need to load a copy of the needle string into memory (so we can search // for it) without leaving it lying around in plaintext in the executable file // as would happen if we used a literal. We might (or might not) leave it // lying around later; that's what the tests are for! const std::string needle = absl::Base64Escape("StrippingTest.Fatal"); EXPECT_DEATH_IF_SUPPORTED(LOG(FATAL) << "U3RyaXBwaW5nVGVzdC5GYXRhbA==", ""); std::unique_ptr> exe = OpenTestExecutable(); ASSERT_THAT(exe, NotNull()); if (absl::LogSeverity::kFatal >= kAbslMinLogLevel) { EXPECT_THAT(exe.get(), FileHasSubstr(needle)); } else { EXPECT_THAT(exe.get(), Not(FileHasSubstr(needle))); } } TEST_F(StrippingTest, Level) { const std::string needle = absl::Base64Escape("StrippingTest.Level"); volatile auto severity = absl::LogSeverity::kWarning; // Ensure that `severity` is not a compile-time constant to prove that // stripping works regardless: LOG(LEVEL(severity)) << "U3RyaXBwaW5nVGVzdC5MZXZlbA=="; std::unique_ptr> exe = OpenTestExecutable(); ASSERT_THAT(exe, NotNull()); if (absl::LogSeverity::kFatal >= kAbslMinLogLevel) { // This can't be stripped at compile-time because it might evaluate to a // level that shouldn't be stripped. EXPECT_THAT(exe.get(), FileHasSubstr(needle)); } else { #if defined(_MSC_VER) || defined(__APPLE__) // Dead code elimination misses this case. #else // All levels should be stripped, so it doesn't matter what the severity // winds up being. EXPECT_THAT(exe.get(), Not(FileHasSubstr(needle))); #endif } } } // namespace