All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.hadoop.hive.common.metrics.metrics2.JsonFileMetricsReporter Maven / Gradle / Ivy

There is a newer version: 4.0.0
Show newest version
/*
 * 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.hadoop.hive.common.metrics.metrics2;

import com.codahale.metrics.MetricRegistry;
import com.codahale.metrics.json.MetricsModule;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.ObjectWriter;
import com.google.common.util.concurrent.ThreadFactoryBuilder;
import org.apache.hadoop.hive.conf.HiveConf;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardCopyOption;
import java.nio.file.attribute.FileAttribute;
import java.nio.file.attribute.PosixFilePermission;
import java.nio.file.attribute.PosixFilePermissions;
import java.util.Set;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;


/**
 * A metrics reporter for CodahaleMetrics that dumps metrics periodically into
 * a file in JSON format. Only files on local filesystems are supported.
 */
public class JsonFileMetricsReporter implements CodahaleReporter, Runnable {
  //
  // Implementation notes.
  //
  // 1. Since only local file systems are supported, there is no need to use Hadoop
  //    version of Path class.
  // 2. java.nio package provides modern implementation of file and directory operations
  //    which is better then the traditional java.io, so we are using it here.
  //    In particular, it supports atomic creation of temporary files with specified
  //    permissions in the specified directory. This also avoids various attacks possible
  //    when temp file name is generated first, followed by file creation.
  //    See http://www.oracle.com/technetwork/articles/javase/nio-139333.html for
  //    the description of NIO API and
  //    http://docs.oracle.com/javase/tutorial/essential/io/legacy.html for the
  //    description of interoperability between legacy IO api vs NIO API.
  // 3. To avoid race conditions with readers of the metrics file, the implementation
  //    dumps metrics to a temporary file in the same directory as the actual metrics
  //    file and then renames it to the destination. Since both are located on the same
  //    filesystem, this rename is likely to be atomic (as long as the underlying OS
  //    support atomic renames.
  //
  // NOTE: This reporter is very similar to
  //       org.apache.hadoop.hive.metastore.metrics.JsonReporter.
  //       It would be good to unify the two.
  //
  private static final Logger LOGGER = LoggerFactory.getLogger(JsonFileMetricsReporter.class);
  // Permissions for the metrics file
  private static final FileAttribute> FILE_ATTRS =
      PosixFilePermissions.asFileAttribute(PosixFilePermissions.fromString("rw-r--r--"));
  // Permissions for metric directory
  private static final FileAttribute> DIR_ATTRS =
      PosixFilePermissions.asFileAttribute(PosixFilePermissions.fromString("rwxr-xr-x"));
  // Thread name for reporter thread
  private static final String JSON_REPORTER_THREAD_NAME = "json-metric-reporter";

  private final MetricRegistry metricRegistry;
  private final ObjectWriter jsonWriter;
  private ScheduledExecutorService executorService;
  private final long interval;
  // Location of JSON file
  private final Path path;
  // Directory where path resides
  private final Path metricsDir;

  public JsonFileMetricsReporter(MetricRegistry registry, HiveConf conf) {
    this.metricRegistry = registry;
    this.jsonWriter =
        new ObjectMapper().registerModule(new MetricsModule(TimeUnit.MILLISECONDS,
            TimeUnit.MILLISECONDS, false)).writerWithDefaultPrettyPrinter();

    interval = conf.getTimeVar(HiveConf.ConfVars.HIVE_METRICS_JSON_FILE_INTERVAL, TimeUnit.MILLISECONDS);
    String pathString = conf.getVar(HiveConf.ConfVars.HIVE_METRICS_JSON_FILE_LOCATION);
    path = Paths.get(pathString).toAbsolutePath();
    LOGGER.info("Reporting metrics to {}", path);
    // We want to use metricsDir in the same directory as the destination file to support atomic
    // move of temp file to the destination metrics file
    metricsDir = path.getParent();
  }

  @Override
  public void start() {
    // Create metrics directory if it is not present
    if (!metricsDir.toFile().exists()) {
      LOGGER.warn("Metrics directory {} does not exist, creating one", metricsDir);
      try {
        // createDirectories creates all non-existent parent directories
        Files.createDirectories(metricsDir, DIR_ATTRS);
      } catch (IOException e) {
        LOGGER.error("Failed to create directory {}: {}", metricsDir, e.getMessage());
        return;
      }
    }

    executorService = Executors.newScheduledThreadPool(1,
        new ThreadFactoryBuilder().setNameFormat(JSON_REPORTER_THREAD_NAME).build());
    executorService.scheduleWithFixedDelay(this,0, interval, TimeUnit.MILLISECONDS);
  }

  @Override
  public void close() {
    if (executorService != null) {
      executorService.shutdown();
    }
  }

  @Override
  public void run() {
    Path tmpFile = null;
    try {
      // Dump metrics to string as JSON
      String json = null;
      try {
        json = jsonWriter.writeValueAsString(metricRegistry);
      } catch (JsonProcessingException e) {
        LOGGER.error("Unable to convert json to string ", e);
        return;
      }

      // Metrics are first dumped to a temp file which is then renamed to the destination
      try {
        tmpFile = Files.createTempFile(metricsDir, "hmetrics", "json", FILE_ATTRS);
      } catch (IOException e) {
        LOGGER.error("failed to create temp file for JSON metrics", e);
        return;
      } catch (SecurityException e) {
        // This shouldn't ever happen
        LOGGER.error("failed to create temp file for JSON metrics: no permissions", e);
        return;
      } catch (UnsupportedOperationException e) {
        // This shouldn't ever happen
        LOGGER.error("failed to create temp file for JSON metrics: operartion not supported", e);
        return;
      }

      // Write json to the temp file.
      try (BufferedWriter bw = new BufferedWriter(new FileWriter(tmpFile.toFile()))) {
        bw.write(json);
      } catch (IOException e) {
        LOGGER.error("Unable to write to temp file " + tmpFile, e);
        return;
      }

      // Atomically move temp file to the destination file
      try {
        Files.move(tmpFile, path, StandardCopyOption.ATOMIC_MOVE);
      } catch (Exception e) {
        LOGGER.error("Unable to rename temp file {} to {}", tmpFile, path);
        LOGGER.error("Exception during rename", e);
      }
    } catch (Throwable t) {
      // catch all errors (throwable and execptions to prevent subsequent tasks from being suppressed)
      LOGGER.error("Error executing scheduled task ", t);
    } finally {
      // If something happened and we were not able to rename the temp file, attempt to remove it
      if (tmpFile != null && tmpFile.toFile().exists()) {
        // Attempt to delete temp file, if this fails, not much can be done about it.
        try {
          Files.delete(tmpFile);
        } catch (Exception e) {
          LOGGER.error("failed to delete temporary metrics file " +  tmpFile, e);
        }
      }
    }
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy