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

org.apache.hadoop.fs.statistics.IOStatisticsSnapshot Maven / Gradle / Ivy

The 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.fs.statistics;

import javax.annotation.Nullable;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.TreeMap;
import java.util.concurrent.ConcurrentHashMap;
import java.util.stream.Collectors;

import com.fasterxml.jackson.annotation.JsonProperty;

import org.apache.hadoop.classification.InterfaceAudience;
import org.apache.hadoop.classification.InterfaceStability;
import org.apache.hadoop.fs.statistics.impl.IOStatisticsBinding;
import org.apache.hadoop.util.JsonSerialization;

import static org.apache.hadoop.util.Preconditions.checkNotNull;
import static org.apache.hadoop.fs.statistics.IOStatisticsLogging.ioStatisticsToString;
import static org.apache.hadoop.fs.statistics.impl.IOStatisticsBinding.aggregateMaps;
import static org.apache.hadoop.fs.statistics.impl.IOStatisticsBinding.snapshotMap;

/**
 * Snapshot of statistics from a different source.
 * 

* It is serializable so that frameworks which can use java serialization * to propagate data (Spark, Flink...) can send the statistics * back. For this reason, TreeMaps are explicitly used as field types, * even though IDEs can recommend use of Map instead. * For security reasons, untrusted java object streams should never be * deserialized. If for some reason this is required, use * {@link #requiredSerializationClasses()} to get the list of classes * used when deserializing instances of this object. *

*

* It is annotated for correct serializations with jackson2. *

*/ @SuppressWarnings("CollectionDeclaredAsConcreteClass") @InterfaceAudience.Public @InterfaceStability.Evolving public final class IOStatisticsSnapshot implements IOStatistics, Serializable, IOStatisticsAggregator, IOStatisticsSetters { private static final long serialVersionUID = -1762522703841538084L; /** * List of chasses needed to deserialize. */ private static final Class[] DESERIALIZATION_CLASSES = { IOStatisticsSnapshot.class, TreeMap.class, Long.class, MeanStatistic.class, }; /** * Counters. */ @JsonProperty private transient Map counters; /** * Gauges. */ @JsonProperty private transient Map gauges; /** * Minimum values. */ @JsonProperty private transient Map minimums; /** * Maximum values. */ @JsonProperty private transient Map maximums; /** * mean statistics. The JSON key is all lower case.. */ @JsonProperty("meanstatistics") private transient Map meanStatistics; /** * Construct. */ public IOStatisticsSnapshot() { createMaps(); } /** * Construct, taking a snapshot of the source statistics data * if the source is non-null. * If the source is null, the empty maps are created * @param source statistics source. Nullable. */ public IOStatisticsSnapshot(IOStatistics source) { if (source != null) { snapshot(source); } else { createMaps(); } } /** * Create the maps. */ private synchronized void createMaps() { counters = new ConcurrentHashMap<>(); gauges = new ConcurrentHashMap<>(); minimums = new ConcurrentHashMap<>(); maximums = new ConcurrentHashMap<>(); meanStatistics = new ConcurrentHashMap<>(); } /** * Clear all the maps. */ public synchronized void clear() { counters.clear(); gauges.clear(); minimums.clear(); maximums.clear(); meanStatistics.clear(); } /** * Take a snapshot. * * This completely overwrites the map data with the statistics * from the source. * @param source statistics source. */ public synchronized void snapshot(IOStatistics source) { checkNotNull(source); counters = snapshotMap(source.counters()); gauges = snapshotMap(source.gauges()); minimums = snapshotMap(source.minimums()); maximums = snapshotMap(source.maximums()); meanStatistics = snapshotMap(source.meanStatistics(), MeanStatistic::copy); } /** * Aggregate the current statistics with the * source reference passed in. * * The operation is synchronized. * @param source source; may be null * @return true if a merge took place. */ @Override public synchronized boolean aggregate( @Nullable IOStatistics source) { if (source == null) { return false; } aggregateMaps(counters, source.counters(), IOStatisticsBinding::aggregateCounters, IOStatisticsBinding::passthroughFn); aggregateMaps(gauges, source.gauges(), IOStatisticsBinding::aggregateGauges, IOStatisticsBinding::passthroughFn); aggregateMaps(minimums, source.minimums(), IOStatisticsBinding::aggregateMinimums, IOStatisticsBinding::passthroughFn); aggregateMaps(maximums, source.maximums(), IOStatisticsBinding::aggregateMaximums, IOStatisticsBinding::passthroughFn); aggregateMaps(meanStatistics, source.meanStatistics(), IOStatisticsBinding::aggregateMeanStatistics, MeanStatistic::copy); return true; } @Override public synchronized Map counters() { return counters; } @Override public synchronized Map gauges() { return gauges; } @Override public synchronized Map minimums() { return minimums; } @Override public synchronized Map maximums() { return maximums; } @Override public synchronized Map meanStatistics() { return meanStatistics; } @Override public synchronized void setCounter(final String key, final long value) { counters().put(key, value); } @Override public synchronized void setGauge(final String key, final long value) { gauges().put(key, value); } @Override public synchronized void setMaximum(final String key, final long value) { maximums().put(key, value); } @Override public synchronized void setMinimum(final String key, final long value) { minimums().put(key, value); } @Override public void setMeanStatistic(final String key, final MeanStatistic value) { meanStatistics().put(key, value); } @Override public String toString() { return ioStatisticsToString(this); } /** * Get a JSON serializer for this class. * @return a serializer. */ public static JsonSerialization serializer() { return new JsonSerialization<>(IOStatisticsSnapshot.class, false, true); } /** * Serialize by converting each map to a TreeMap, and saving that * to the stream. * @param s ObjectOutputStream. * @throws IOException raised on errors performing I/O. */ private synchronized void writeObject(ObjectOutputStream s) throws IOException { // Write out the core s.defaultWriteObject(); s.writeObject(new TreeMap(counters)); s.writeObject(new TreeMap(gauges)); s.writeObject(new TreeMap(minimums)); s.writeObject(new TreeMap(maximums)); s.writeObject(new TreeMap(meanStatistics)); } /** * Deserialize by loading each TreeMap, and building concurrent * hash maps from them. * * @param s ObjectInputStream. * @throws IOException raised on errors performing I/O. * @throws ClassNotFoundException class not found exception */ private void readObject(final ObjectInputStream s) throws IOException, ClassNotFoundException { // read in core s.defaultReadObject(); // and rebuild a concurrent hashmap from every serialized tree map // read back from the stream. counters = new ConcurrentHashMap<>( (TreeMap) s.readObject()); gauges = new ConcurrentHashMap<>( (TreeMap) s.readObject()); minimums = new ConcurrentHashMap<>( (TreeMap) s.readObject()); maximums = new ConcurrentHashMap<>( (TreeMap) s.readObject()); meanStatistics = new ConcurrentHashMap<>( (TreeMap) s.readObject()); } /** * What classes are needed to deserialize this class? * Needed to securely unmarshall this from untrusted sources. * @return a list of required classes to deserialize the data. */ public static List requiredSerializationClasses() { return Arrays.stream(DESERIALIZATION_CLASSES) .collect(Collectors.toList()); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy