| // Copyright 2019 The Fuchsia Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <fidl/fuchsia.boot/cpp/fidl.h> |
| #include <lib/async/cpp/executor.h> |
| #include <lib/component/incoming/cpp/service_client.h> |
| #include <lib/fdio/directory.h> |
| #include <lib/fidl/cpp/binding_set.h> |
| #include <lib/inspect/cpp/vmo/types.h> |
| #include <lib/syslog/cpp/macros.h> |
| #include <lib/zx/time.h> |
| #include <zircon/errors.h> |
| |
| #include <memory> |
| #include <string_view> |
| #include <vector> |
| |
| #include <gmock/gmock.h> |
| #include <gtest/gtest.h> |
| |
| #include "src/developer/forensics/feedback/attachments/kernel_log.h" |
| #include "src/developer/forensics/feedback/attachments/types.h" |
| #include "src/developer/forensics/testing/gmatchers.h" |
| #include "src/lib/fxl/strings/string_printf.h" |
| #include "src/lib/testing/loop_fixture/real_loop_fixture.h" |
| #include "src/lib/testing/predicates/status.h" |
| |
| namespace forensics::feedback { |
| namespace { |
| |
| using testing::UnorderedElementsAreArray; |
| |
| class CollectKernelLogTest : public gtest::RealLoopFixture { |
| public: |
| CollectKernelLogTest() : executor_(dispatcher()) { |
| environment_services_ = sys::ServiceDirectory::CreateFromNamespace(); |
| redactor_ = std::unique_ptr<RedactorBase>(new IdentityRedactor(inspect::BoolProperty())); |
| } |
| |
| void SetRedactor(std::unique_ptr<RedactorBase> redactor) { redactor_ = std::move(redactor); } |
| |
| AttachmentValue GetKernelLog() { |
| const uint64_t kTicket = 1234; |
| KernelLog kernel_log(dispatcher(), environment_services_, nullptr, redactor_.get()); |
| ::fpromise::result<AttachmentValue> attachment(::fpromise::error()); |
| executor_.schedule_task(kernel_log.Get(kTicket) |
| .and_then([&attachment](AttachmentValue& result) { |
| attachment = ::fpromise::ok(std::move(result)); |
| }) |
| .or_else([] { FX_LOGS(FATAL) << "Unreachable branch"; })); |
| |
| RunLoopUntil([&attachment] { return attachment.is_ok(); }); |
| return attachment.take_value(); |
| } |
| |
| protected: |
| async::Executor& GetExecutor() { return executor_; } |
| |
| protected: |
| std::shared_ptr<sys::ServiceDirectory> environment_services_; |
| async::Executor executor_; |
| std::unique_ptr<RedactorBase> redactor_; |
| }; |
| |
| void SendToKernelLog(std::string_view str) { |
| auto client_end = component::Connect<fuchsia_boot::WriteOnlyLog>(); |
| ASSERT_OK(client_end.status_value()); |
| |
| auto result = fidl::WireSyncClient(std::move(*client_end))->Get(); |
| ASSERT_OK(result.status()); |
| zx::debuglog log = std::move(result->log); |
| |
| zx_debuglog_write(log.get(), 0, str.data(), str.size()); |
| } |
| |
| TEST_F(CollectKernelLogTest, Succeed_BasicCase) { |
| const std::string output( |
| fxl::StringPrintf("<<GetLogTest_Succeed_BasicCase: %zu>>", zx_clock_get_monotonic())); |
| SendToKernelLog(output); |
| |
| const auto log = GetKernelLog(); |
| |
| ASSERT_TRUE(log.HasValue()); |
| EXPECT_THAT(log.Value(), testing::HasSubstr(output)); |
| } |
| |
| TEST_F(CollectKernelLogTest, GetTerminatesDueToForceCompletion) { |
| const std::string output(fxl::StringPrintf( |
| "<<GetLogTest_Get_Terminates_Due_To_ForceCompletion: %zu>>", zx_clock_get_monotonic())); |
| const uint64_t kTicket = 1234; |
| |
| SendToKernelLog(output); |
| |
| AttachmentValue log(Error::kNotSet); |
| ::fpromise::result<AttachmentValue> attachment(::fpromise::error()); |
| |
| KernelLog kernel_log(dispatcher(), environment_services_, nullptr, redactor_.get()); |
| GetExecutor().schedule_task(kernel_log.Get(kTicket).and_then( |
| [&attachment](AttachmentValue& result) { attachment = ::fpromise::ok(std::move(result)); })); |
| kernel_log.ForceCompletion(kTicket, Error::kDefault); |
| |
| RunLoopUntil([&attachment] { return attachment.is_ok(); }); |
| log = attachment.take_value(); |
| |
| EXPECT_THAT(log, AttachmentValueIs(Error::kDefault)); |
| } |
| |
| TEST_F(CollectKernelLogTest, ForceCompletionCalledAfterTermination) { |
| const std::string output(fxl::StringPrintf( |
| "<<GetLogTest_ForceCompletion_Called_After_Termination: %zu>>", zx_clock_get_monotonic())); |
| const uint64_t kTicket = 1234; |
| |
| SendToKernelLog(output); |
| |
| AttachmentValue log(Error::kNotSet); |
| ::fpromise::result<AttachmentValue> attachment(::fpromise::error()); |
| |
| KernelLog kernel_log(dispatcher(), environment_services_, nullptr, redactor_.get()); |
| GetExecutor().schedule_task(kernel_log.Get(kTicket).and_then( |
| [&attachment](AttachmentValue& result) { attachment = ::fpromise::ok(std::move(result)); })); |
| |
| RunLoopUntil([&attachment] { return attachment.is_ok(); }); |
| log = attachment.take_value(); |
| |
| kernel_log.ForceCompletion(kTicket, Error::kDefault); |
| |
| ASSERT_FALSE(log.HasError()); |
| |
| ASSERT_TRUE(log.HasValue()); |
| EXPECT_THAT(log.Value(), testing::HasSubstr(output)); |
| } |
| |
| TEST_F(CollectKernelLogTest, GetCalledWithSameTicket) { |
| const uint64_t kTicket = 1234; |
| |
| KernelLog kernel_log(dispatcher(), environment_services_, nullptr, redactor_.get()); |
| |
| // Expect a crash because a ticket cannot be reused. |
| ASSERT_DEATH( |
| { |
| const auto log1 = kernel_log.Get(kTicket); |
| const auto log2 = kernel_log.Get(kTicket); |
| }, |
| "Ticket used twice: "); |
| } |
| |
| TEST_F(CollectKernelLogTest, Succeed_TwoRetrievals) { |
| // ReadOnlyLog was returning a shared handle so the second reader would get data after where the |
| // first had read from. Confirm that both readers get the target string. |
| const std::string output( |
| fxl::StringPrintf("<<GetLogTest_Succeed_TwoRetrievals: %zu>>", zx_clock_get_monotonic())); |
| SendToKernelLog(output); |
| |
| const auto log1 = GetKernelLog(); |
| |
| ASSERT_TRUE(log1.HasValue()); |
| EXPECT_THAT(log1.Value(), testing::HasSubstr(output)); |
| |
| const auto log2 = GetKernelLog(); |
| |
| ASSERT_TRUE(log2.HasValue()); |
| EXPECT_THAT(log2.Value(), testing::HasSubstr(output)); |
| } |
| |
| class SimpleRedactor : public RedactorBase { |
| public: |
| SimpleRedactor() : RedactorBase(inspect::BoolProperty()) {} |
| |
| std::string& Redact(std::string& text) override { |
| text = "<REDACTED>"; |
| return text; |
| } |
| |
| std::string UnredactedCanary() const override { return ""; } |
| std::string RedactedCanary() const override { return ""; } |
| }; |
| |
| TEST_F(CollectKernelLogTest, Succeed_Redacts) { |
| SetRedactor(std::make_unique<SimpleRedactor>()); |
| const std::string output( |
| fxl::StringPrintf("<<GetLogTest_Succeed_BasicCase: %zu>>", zx_clock_get_monotonic())); |
| SendToKernelLog(output); |
| |
| const auto log = GetKernelLog(); |
| |
| ASSERT_TRUE(log.HasValue()); |
| EXPECT_THAT(log.Value(), testing::HasSubstr("<REDACTED>")); |
| } |
| |
| } // namespace |
| } // namespace forensics::feedback |