You are viewing a plain text version of this content. The canonical link for it is here.
Posted to commits@samza.apache.org by ja...@apache.org on 2019/01/17 22:06:07 UTC

samza git commit: DiagnosticsAppender for log4j2

Repository: samza
Updated Branches:
  refs/heads/master 193fd272c -> cfedd5971


DiagnosticsAppender for log4j2

This PR adds a DiagnosticsAppender for log4j2.
There exists one already for log4j in samza-log4j.

Two points to note
1. The Appender, LogEvent, and Configuration APIs between log4j2 and log4j are completely different.

2. Log4j requires you to extend the AppenderSkeleton, while log4j2 requires extending the AbstractAppender.

So there was very little overlap (and very little value) in creating a base-class for the two appenders, specially given that java doesn't allow multi-inheritance.

Author: Ray Matharu <rm...@linkedin.com>

Reviewers: Pawas C<pc...@linkedin.com>

Closes #882 from rmatharu/log4j2appender


Project: http://git-wip-us.apache.org/repos/asf/samza/repo
Commit: http://git-wip-us.apache.org/repos/asf/samza/commit/cfedd597
Tree: http://git-wip-us.apache.org/repos/asf/samza/tree/cfedd597
Diff: http://git-wip-us.apache.org/repos/asf/samza/diff/cfedd597

Branch: refs/heads/master
Commit: cfedd59715260d8f2d2259ddad2ee1e1f7f28944
Parents: 193fd27
Author: Ray Matharu <rm...@linkedin.com>
Authored: Thu Jan 17 14:06:00 2019 -0800
Committer: Jagadish <jv...@linkedin.com>
Committed: Thu Jan 17 14:06:00 2019 -0800

----------------------------------------------------------------------
 .../log4j2/SimpleDiagnosticsAppender.java       | 95 ++++++++++++++++++++
 1 file changed, 95 insertions(+)
----------------------------------------------------------------------


http://git-wip-us.apache.org/repos/asf/samza/blob/cfedd597/samza-log4j2/src/main/java/org/apache/samza/logging/log4j2/SimpleDiagnosticsAppender.java
----------------------------------------------------------------------
diff --git a/samza-log4j2/src/main/java/org/apache/samza/logging/log4j2/SimpleDiagnosticsAppender.java b/samza-log4j2/src/main/java/org/apache/samza/logging/log4j2/SimpleDiagnosticsAppender.java
new file mode 100644
index 0000000..37133db
--- /dev/null
+++ b/samza-log4j2/src/main/java/org/apache/samza/logging/log4j2/SimpleDiagnosticsAppender.java
@@ -0,0 +1,95 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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.
+ */
+
+package org.apache.samza.logging.log4j2;
+
+import org.apache.logging.log4j.LogManager;
+import org.apache.logging.log4j.core.Appender;
+import org.apache.logging.log4j.core.LogEvent;
+import org.apache.logging.log4j.core.LoggerContext;
+import org.apache.logging.log4j.core.appender.AbstractAppender;
+import org.apache.logging.log4j.core.config.Configuration;
+import org.apache.logging.log4j.core.config.LoggerConfig;
+import org.apache.samza.container.SamzaContainerMetrics;
+import org.apache.samza.diagnostics.DiagnosticsExceptionEvent;
+import org.apache.samza.metrics.ListGauge;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Provides an in-memory appender that parses LogEvents to filter events relevant to diagnostics.
+ * Currently, filters exception related events and update an exception metric ({@link ListGauge}) in
+ * {@link SamzaContainerMetrics}.
+ *
+ * When used inconjunction with {@link org.apache.samza.metrics.reporter.MetricsSnapshotReporter} provides a
+ * stream of diagnostics-related events.
+ */
+public class SimpleDiagnosticsAppender extends AbstractAppender {
+  private static final Logger LOG = LoggerFactory.getLogger(SimpleDiagnosticsAppender.class);
+
+  // simple object to synchronize root logger attachment
+  private static final Object SYNCHRONIZATION_OBJECT = new Object();
+  protected final ListGauge<DiagnosticsExceptionEvent> samzaContainerExceptionMetric;
+
+  public SimpleDiagnosticsAppender(SamzaContainerMetrics samzaContainerMetrics) {
+    super(SimpleDiagnosticsAppender.class.getName(), null, null);
+    this.samzaContainerExceptionMetric = samzaContainerMetrics.exceptions();
+
+    synchronized (SYNCHRONIZATION_OBJECT) {
+      attachAppenderToLoggers(this);
+    }
+
+    System.out.println("SimpleDiagnosticsAppender initialized ");
+  }
+
+  private void attachAppenderToLoggers(Appender appender) {
+    LoggerContext ctx = (LoggerContext) LogManager.getContext(false);
+    Configuration config = ctx.getConfiguration();
+
+    // ensure appender is attached only once per JVM (regardless of #containers)
+    if (config.getRootLogger().getAppenders().get(SimpleDiagnosticsAppender.class.getName()) == null) {
+      LOG.info("Attaching diagnostics appender to root logger");
+      appender.start();
+      config.addAppender(appender);
+      for (final LoggerConfig loggerConfig : config.getLoggers().values()) {
+        loggerConfig.addAppender(appender, null, null);
+      }
+    }
+  }
+
+  @Override
+  public void append(LogEvent logEvent) {
+    try {
+      // if an event with a non-null throwable is received => exception event
+      if (logEvent.getThrown() != null) {
+        DiagnosticsExceptionEvent diagnosticsExceptionEvent =
+            new DiagnosticsExceptionEvent(logEvent.getTimeMillis(), logEvent.getThrown(),
+                logEvent.getContextData().toMap());
+
+        samzaContainerExceptionMetric.add(diagnosticsExceptionEvent);
+        LOG.debug("Received DiagnosticsExceptionEvent " + diagnosticsExceptionEvent);
+      } else {
+        LOG.debug("Received non-exception event with message " + logEvent.getMessage());
+      }
+    } catch (Exception e) {
+      // blanket catch of all exceptions so as to not impact any job
+      LOG.error("Exception in logevent parsing", e);
+    }
+  }
+}