org.glowroot.collector.AggregateBuilder.java Source code

Java tutorial

Introduction

Here is the source code for org.glowroot.collector.AggregateBuilder.java

Source

/*
 * Copyright 2013-2015 the original author or authors.
 *
 * Licensed 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.glowroot.collector;

import java.io.IOException;
import java.nio.ByteBuffer;
import java.util.Map;

import javax.annotation.Nullable;

import com.fasterxml.jackson.core.JsonFactory;
import com.fasterxml.jackson.core.JsonGenerator;
import com.google.common.collect.Maps;
import com.google.common.io.CharStreams;

import org.glowroot.common.ScratchBuffer;
import org.glowroot.transaction.model.Profile;
import org.glowroot.transaction.model.ThreadInfoData;
import org.glowroot.transaction.model.TimerImpl;
import org.glowroot.transaction.model.Transaction;

import static java.util.concurrent.TimeUnit.NANOSECONDS;

// must be used under an appropriate lock
class AggregateBuilder {

    private static final JsonFactory jsonFactory = new JsonFactory();

    private final String transactionType;
    private final @Nullable String transactionName;
    private long totalMicros;
    private long errorCount;
    private long transactionCount;
    private @Nullable Long totalCpuTime;
    private @Nullable Long totalBlockedTime;
    private @Nullable Long totalWaitedTime;
    private @Nullable Long totalAllocatedBytes;
    private long profileSampleCount;
    private long traceCount;
    // histogram uses microseconds to reduce (or at least simplify) bucket allocations
    private final LazyHistogram histogram = new LazyHistogram();
    private final AggregateTimer syntheticRootTimer = new AggregateTimer("");
    private final AggregateProfileBuilder aggregateProfile = new AggregateProfileBuilder();

    AggregateBuilder(String transactionType, @Nullable String transactionName) {
        this.transactionType = transactionType;
        this.transactionName = transactionName;
    }

    void add(Transaction transaction) {
        long durationMicros = NANOSECONDS.toMicros(transaction.getDuration());
        totalMicros += durationMicros;
        if (transaction.getErrorMessage() != null) {
            errorCount++;
        }
        if (transaction.willBeStored()) {
            traceCount++;
        }
        transactionCount++;
        ThreadInfoData threadInfo = transaction.getThreadInfo();
        if (threadInfo != null) {
            totalCpuTime = nullAwareAdd(totalCpuTime, threadInfo.threadCpuTime());
            totalBlockedTime = nullAwareAdd(totalBlockedTime, threadInfo.threadBlockedTime());
            totalWaitedTime = nullAwareAdd(totalWaitedTime, threadInfo.threadWaitedTime());
            totalAllocatedBytes = nullAwareAdd(totalAllocatedBytes, threadInfo.threadAllocatedBytes());
        }
        histogram.add(durationMicros);
    }

    void addToTimers(TimerImpl rootTimer) {
        syntheticRootTimer.totalMicros += NANOSECONDS.toMicros(rootTimer.getTotal());
        syntheticRootTimer.count += rootTimer.getCount();
        mergeAsChildTimer(syntheticRootTimer, rootTimer);
    }

    void addToProfile(Profile profile) {
        aggregateProfile.addProfile(profile);
        profileSampleCount += profile.getSampleCount();
    }

    Aggregate build(long captureTime, ScratchBuffer scratchBuffer) throws IOException {
        ByteBuffer buffer = scratchBuffer.getBuffer(histogram.getNeededByteBufferCapacity());
        buffer.clear();
        byte[] histogramBytes = histogram.encodeUsingTempByteBuffer(buffer);
        return Aggregate.builder().transactionType(transactionType).transactionName(transactionName)
                .captureTime(captureTime).totalMicros(totalMicros).errorCount(errorCount)
                .transactionCount(transactionCount).totalCpuMicros(nullAwareNanosToMicros(totalCpuTime))
                .totalBlockedMicros(nullAwareNanosToMicros(totalBlockedTime))
                .totalWaitedMicros(nullAwareNanosToMicros(totalWaitedTime))
                .totalAllocatedKBytes(nullAwareBytesToKBytes(totalAllocatedBytes)).timers(getTimersJson())
                .histogram(histogramBytes).profileSampleCount(profileSampleCount).traceCount(traceCount)
                .profile(getProfileJson()).build();
    }

    TransactionSummary getLiveTransactionSummary() {
        return TransactionSummary.builder().transactionName(transactionName).totalMicros(totalMicros)
                .transactionCount(transactionCount).build();
    }

    ErrorSummary getLiveErrorSummary() {
        return ErrorSummary.builder().transactionName(transactionName).errorCount(errorCount)
                .transactionCount(transactionCount).build();
    }

    @Nullable
    ErrorPoint buildErrorPoint(long captureTime) {
        if (errorCount == 0) {
            return null;
        }
        return ErrorPoint.builder().captureTime(captureTime).errorCount(errorCount)
                .transactionCount(transactionCount).build();
    }

    long getProfileSampleCount() {
        return profileSampleCount;
    }

    private void mergeAsChildTimer(AggregateTimer parentAggregateTimer, TimerImpl timer) {
        String name = timer.getName();
        AggregateTimer aggregateTimer = parentAggregateTimer.nestedTimers.get(name);
        if (aggregateTimer == null) {
            aggregateTimer = new AggregateTimer(name);
            parentAggregateTimer.nestedTimers.put(name, aggregateTimer);
        }
        aggregateTimer.totalMicros += NANOSECONDS.toMicros(timer.getTotal());
        aggregateTimer.count += timer.getCount();
        for (TimerImpl nestedTimer : timer.getNestedTimers()) {
            mergeAsChildTimer(aggregateTimer, nestedTimer);
        }
    }

    private String getTimersJson() throws IOException {
        StringBuilder sb = new StringBuilder();
        JsonGenerator jg = jsonFactory.createGenerator(CharStreams.asWriter(sb));
        writeTimer(jg, syntheticRootTimer);
        jg.close();
        return sb.toString();
    }

    @Nullable
    String getProfileJson() throws IOException {
        return ProfileCharSourceCreator.createProfileJson(aggregateProfile.getSyntheticRootNode());
    }

    private void writeTimer(JsonGenerator jg, AggregateTimer aggregateTimer) throws IOException {
        jg.writeStartObject();
        jg.writeStringField("name", aggregateTimer.name);
        jg.writeNumberField("totalMicros", aggregateTimer.totalMicros);
        jg.writeNumberField("count", aggregateTimer.count);
        if (!aggregateTimer.nestedTimers.isEmpty()) {
            jg.writeArrayFieldStart("nestedTimers");
            for (AggregateTimer timer : aggregateTimer.nestedTimers.values()) {
                writeTimer(jg, timer);
            }
            jg.writeEndArray();
        }
        jg.writeEndObject();
    }

    private static @Nullable Long nullAwareNanosToMicros(@Nullable Long nanoseconds) {
        if (nanoseconds == null) {
            return null;
        }
        return NANOSECONDS.toMicros(nanoseconds);
    }

    private static @Nullable Long nullAwareBytesToKBytes(@Nullable Long bytes) {
        if (bytes == null) {
            return null;
        }
        return bytes / 1024;
    }

    private static @Nullable Long nullAwareAdd(@Nullable Long x, @Nullable Long y) {
        if (x == null) {
            return y;
        }
        if (y == null) {
            return x;
        }
        return x + y;
    }

    private static class AggregateTimer {
        private final String name;
        // aggregation uses microseconds to avoid (unlikely) 292 year nanosecond rollover
        private long totalMicros;
        private long count;
        private final Map<String, AggregateTimer> nestedTimers = Maps.newHashMap();

        private AggregateTimer(String name) {
            this.name = name;
        }
    }
}