Skip to content

Commit

Permalink
feat: logging can be enabled via env var (googleapis/google-cloud-cpp…
Browse files Browse the repository at this point in the history
…-common#181)

Users can now enable logging to `std::clog` without any code changes or
recompiling their code simply by setting the
"GOOGLE_CLOUD_CPP_ENABLE_CLOG" environment variable before running their
program.

This env var is currently being inspected and set in some of our other
code, such as
https://github.com/googleapis/google-cloud-cpp-spanner/blob/20e2f9b011e3747fe5a8b8dc284245c11ff354b6/google/cloud/spanner/connection_options.cc#L68,
and we should remove that after this change is merged.
  • Loading branch information
devjgm authored Feb 20, 2020
1 parent cf39626 commit 0e4d436
Show file tree
Hide file tree
Showing 6 changed files with 50 additions and 34 deletions.
7 changes: 0 additions & 7 deletions google/cloud/connection_options.cc
Original file line number Diff line number Diff line change
Expand Up @@ -42,13 +42,6 @@ TracingOptions DefaultTracingOptions() {
return TracingOptions{}.SetOptions(*tracing_options);
}

void DefaultLogging() {
if (google::cloud::internal::GetEnv("GOOGLE_CLOUD_CPP_ENABLE_CLOG")
.has_value()) {
google::cloud::LogSink::EnableStdClog();
}
}

std::unique_ptr<BackgroundThreads> DefaultBackgroundThreads() {
return google::cloud::internal::make_unique<
AutomaticallyCreatedBackgroundThreads>();
Expand Down
5 changes: 1 addition & 4 deletions google/cloud/connection_options.h
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,6 @@ inline namespace GOOGLE_CLOUD_CPP_NS {
namespace internal {
std::set<std::string> DefaultTracingComponents();
TracingOptions DefaultTracingOptions();
void DefaultLogging();
std::unique_ptr<BackgroundThreads> DefaultBackgroundThreads();
} // namespace internal

Expand All @@ -53,9 +52,7 @@ class ConnectionOptions {
tracing_components_(internal::DefaultTracingComponents()),
tracing_options_(internal::DefaultTracingOptions()),
user_agent_prefix_(ConnectionTraits::user_agent_prefix()),
background_threads_factory_(internal::DefaultBackgroundThreads) {
internal::DefaultLogging();
}
background_threads_factory_(internal::DefaultBackgroundThreads) {}

/// Change the gRPC credentials value.
ConnectionOptions& set_credentials(
Expand Down
19 changes: 0 additions & 19 deletions google/cloud/connection_options_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -233,25 +233,6 @@ TEST(ConnectionOptionsTest, DefaultTracingOptionsWithValue) {
EXPECT_EQ(42, actual.truncate_string_field_longer_than());
}

TEST(ConnectionOptionsTest, DefaultLoggingNoEnvironment) {
EnvironmentVariableRestore restore("GOOGLE_CLOUD_CPP_ENABLE_CLOG");
LogSink::Instance().ClearBackends();
internal::UnsetEnv("GOOGLE_CLOUD_CPP_ENABLE_CLOG");
internal::DefaultLogging();
EXPECT_EQ(0, LogSink::Instance().BackendCount());
}

TEST(ConnectionOptionsTest, DefaultLoggingWithValue) {
EnvironmentVariableRestore restore("GOOGLE_CLOUD_CPP_ENABLE_CLOG");
LogSink::Instance().ClearBackends();
EXPECT_EQ(0, LogSink::Instance().BackendCount());
internal::SetEnv("GOOGLE_CLOUD_CPP_ENABLE_CLOG", "any-value");
internal::DefaultLogging();
EXPECT_EQ(1, LogSink::Instance().BackendCount());
LogSink::DisableStdClog();
EXPECT_EQ(0, LogSink::Instance().BackendCount());
}

TEST(ConnectionOptionsTest, DefaultBackgroundThreads) {
auto actual = internal::DefaultBackgroundThreads();
EXPECT_TRUE(actual);
Expand Down
11 changes: 9 additions & 2 deletions google/cloud/log.cc
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
// limitations under the License.

#include "google/cloud/log.h"
#include "google/cloud/internal/getenv.h"

namespace google {
namespace cloud {
Expand Down Expand Up @@ -45,8 +46,14 @@ LogSink::LogSink()
clog_backend_id_(0) {}

LogSink& LogSink::Instance() {
static LogSink instance;
return instance;
static auto* const kInstance = [] {
auto* p = new LogSink;
if (internal::GetEnv("GOOGLE_CLOUD_CPP_ENABLE_CLOG").has_value()) {
p->EnableStdClogImpl();
}
return p;
}();
return *kInstance;
}

long LogSink::AddBackend(std::shared_ptr<LogBackend> backend) {
Expand Down
12 changes: 11 additions & 1 deletion google/cloud/log.h
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,11 @@
* }
* @endcode
*
* Alternatively, the application can enable logging to `std::clog` without any
* code changes or recompiling by setting the "GOOGLE_CLOUD_CPP_ENABLE_CLOG"
* environment variable before the program starts. The existence of this
* variable is all that matters; the value is ignored.
*
* Note that while `std::clog` is buffered, the framework will flush any log
* message at severity `WARNING` or higher.
*
Expand Down Expand Up @@ -297,7 +302,12 @@ class LogSink {

void Log(LogRecord log_record);

/// Enable `std::clog` on `LogSink::Instance()`.
/**
* Enable `std::clog` on `LogSink::Instance()`.
*
* This is also enabled if the "GOOGLE_CLOUD_CPP_ENABLE_CLOG" environment
* variable is set.
*/
static void EnableStdClog() { Instance().EnableStdClogImpl(); }

/// Disable `std::clog` on `LogSink::Instance()`.
Expand Down
30 changes: 29 additions & 1 deletion google/cloud/log_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -13,14 +13,19 @@
// limitations under the License.

#include "google/cloud/log.h"
#include "google/cloud/internal/setenv.h"
#include "google/cloud/testing_util/environment_variable_restore.h"
#include <gmock/gmock.h>

namespace google {
namespace cloud {
inline namespace GOOGLE_CLOUD_CPP_NS {
namespace {

using namespace ::testing;
using ::testing::_;
using ::testing::ExitedWithCode;
using ::testing::HasSubstr;
using ::testing::Invoke;

TEST(LogSeverityTest, Streaming) {
std::ostringstream os;
Expand Down Expand Up @@ -140,6 +145,29 @@ TEST(LogSinkTest, ClogMultiple) {
EXPECT_EQ(0, LogSink::Instance().BackendCount());
}

TEST(LogSinkTest, ClogEnvironment) {
// We set the death test style to "threadsafe", which causes the
// ASSERT_EXIT() call to re-exec the test binary before executing the given
// statement. This makes the death test thread-safe and it also ensures that
// the LogSink singleton instance will be reconstructed in the child and will
// see the environment variable. See also:
// https://github.com/google/googletest/blob/master/googletest/docs/advanced.md#death-test-styles
auto old_style = testing::FLAGS_gtest_death_test_style;
testing::FLAGS_gtest_death_test_style = "threadsafe";

testing_util::EnvironmentVariableRestore restore(
"GOOGLE_CLOUD_CPP_ENABLE_CLOG");
internal::SetEnv("GOOGLE_CLOUD_CPP_ENABLE_CLOG", "anyvalue");

auto f = [] {
GCP_LOG(INFO) << "testing clog";
std::exit(42);
};
ASSERT_EXIT(f(), ExitedWithCode(42), HasSubstr("testing clog"));

testing::FLAGS_gtest_death_test_style = old_style;
}

namespace {
/// A class to count calls to IOStream operator.
struct IOStreamCounter {
Expand Down

0 comments on commit 0e4d436

Please sign in to comment.