/* * 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 com.facebook.presto.operator; import com.facebook.presto.sql.planner.plan.PlanNodeId; import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonProperty; import com.google.common.collect.ImmutableList; import io.airlift.units.DataSize; import io.airlift.units.Duration; import javax.annotation.Nullable; import javax.annotation.concurrent.Immutable; import java.util.Optional; import static com.google.common.base.Preconditions.checkArgument; import static io.airlift.units.DataSize.succinctBytes; import static java.util.Objects.requireNonNull; import static java.util.concurrent.TimeUnit.NANOSECONDS; @Immutable public class OperatorStats { private final int pipelineId; private final int operatorId; private final PlanNodeId planNodeId; private final String operatorType; private final long totalDrivers; private final long addInputCalls; private final Duration addInputWall; private final Duration addInputCpu; private final Duration addInputUser; private final DataSize inputDataSize; private final long inputPositions; private final double sumSquaredInputPositions; private final long getOutputCalls; private final Duration getOutputWall; private final Duration getOutputCpu; private final Duration getOutputUser; private final DataSize outputDataSize; private final long outputPositions; private final Duration blockedWall; private final long finishCalls; private final Duration finishWall; private final Duration finishCpu; private final Duration finishUser; private final DataSize memoryReservation; private final DataSize systemMemoryReservation; private final Optional<BlockedReason> blockedReason; private final OperatorInfo info; @JsonCreator public OperatorStats( @JsonProperty("pipelineId") int pipelineId, @JsonProperty("operatorId") int operatorId, @JsonProperty("planNodeId") PlanNodeId planNodeId, @JsonProperty("operatorType") String operatorType, @JsonProperty("totalDrivers") long totalDrivers, @JsonProperty("addInputCalls") long addInputCalls, @JsonProperty("addInputWall") Duration addInputWall, @JsonProperty("addInputCpu") Duration addInputCpu, @JsonProperty("addInputUser") Duration addInputUser, @JsonProperty("inputDataSize") DataSize inputDataSize, @JsonProperty("inputPositions") long inputPositions, @JsonProperty("sumSquaredInputPositions") double sumSquaredInputPositions, @JsonProperty("getOutputCalls") long getOutputCalls, @JsonProperty("getOutputWall") Duration getOutputWall, @JsonProperty("getOutputCpu") Duration getOutputCpu, @JsonProperty("getOutputUser") Duration getOutputUser, @JsonProperty("outputDataSize") DataSize outputDataSize, @JsonProperty("outputPositions") long outputPositions, @JsonProperty("blockedWall") Duration blockedWall, @JsonProperty("finishCalls") long finishCalls, @JsonProperty("finishWall") Duration finishWall, @JsonProperty("finishCpu") Duration finishCpu, @JsonProperty("finishUser") Duration finishUser, @JsonProperty("memoryReservation") DataSize memoryReservation, @JsonProperty("systemMemoryReservation") DataSize systemMemoryReservation, @JsonProperty("blockedReason") Optional<BlockedReason> blockedReason, @JsonProperty("info") OperatorInfo info) { this.pipelineId = pipelineId; checkArgument(operatorId >= 0, "operatorId is negative"); this.operatorId = operatorId; this.planNodeId = requireNonNull(planNodeId, "planNodeId is null"); this.operatorType = requireNonNull(operatorType, "operatorType is null"); this.totalDrivers = totalDrivers; this.addInputCalls = addInputCalls; this.addInputWall = requireNonNull(addInputWall, "addInputWall is null"); this.addInputCpu = requireNonNull(addInputCpu, "addInputCpu is null"); this.addInputUser = requireNonNull(addInputUser, "addInputUser is null"); this.inputDataSize = requireNonNull(inputDataSize, "inputDataSize is null"); checkArgument(inputPositions >= 0, "inputPositions is negative"); this.inputPositions = inputPositions; this.sumSquaredInputPositions = sumSquaredInputPositions; this.getOutputCalls = getOutputCalls; this.getOutputWall = requireNonNull(getOutputWall, "getOutputWall is null"); this.getOutputCpu = requireNonNull(getOutputCpu, "getOutputCpu is null"); this.getOutputUser = requireNonNull(getOutputUser, "getOutputUser is null"); this.outputDataSize = requireNonNull(outputDataSize, "outputDataSize is null"); checkArgument(outputPositions >= 0, "outputPositions is negative"); this.outputPositions = outputPositions; this.blockedWall = requireNonNull(blockedWall, "blockedWall is null"); this.finishCalls = finishCalls; this.finishWall = requireNonNull(finishWall, "finishWall is null"); this.finishCpu = requireNonNull(finishCpu, "finishCpu is null"); this.finishUser = requireNonNull(finishUser, "finishUser is null"); this.memoryReservation = requireNonNull(memoryReservation, "memoryReservation is null"); this.systemMemoryReservation = requireNonNull(systemMemoryReservation, "systemMemoryReservation is null"); this.blockedReason = blockedReason; this.info = info; } @JsonProperty public int getPipelineId() { return pipelineId; } @JsonProperty public int getOperatorId() { return operatorId; } @JsonProperty public PlanNodeId getPlanNodeId() { return planNodeId; } @JsonProperty public String getOperatorType() { return operatorType; } @JsonProperty public long getTotalDrivers() { return totalDrivers; } @JsonProperty public long getAddInputCalls() { return addInputCalls; } @JsonProperty public Duration getAddInputWall() { return addInputWall; } @JsonProperty public Duration getAddInputCpu() { return addInputCpu; } @JsonProperty public Duration getAddInputUser() { return addInputUser; } @JsonProperty public DataSize getInputDataSize() { return inputDataSize; } @JsonProperty public long getInputPositions() { return inputPositions; } @JsonProperty public double getSumSquaredInputPositions() { return sumSquaredInputPositions; } @JsonProperty public long getGetOutputCalls() { return getOutputCalls; } @JsonProperty public Duration getGetOutputWall() { return getOutputWall; } @JsonProperty public Duration getGetOutputCpu() { return getOutputCpu; } @JsonProperty public Duration getGetOutputUser() { return getOutputUser; } @JsonProperty public DataSize getOutputDataSize() { return outputDataSize; } @JsonProperty public long getOutputPositions() { return outputPositions; } @JsonProperty public Duration getBlockedWall() { return blockedWall; } @JsonProperty public long getFinishCalls() { return finishCalls; } @JsonProperty public Duration getFinishWall() { return finishWall; } @JsonProperty public Duration getFinishCpu() { return finishCpu; } @JsonProperty public Duration getFinishUser() { return finishUser; } @JsonProperty public DataSize getMemoryReservation() { return memoryReservation; } @JsonProperty public DataSize getSystemMemoryReservation() { return systemMemoryReservation; } @JsonProperty public Optional<BlockedReason> getBlockedReason() { return blockedReason; } @Nullable @JsonProperty public OperatorInfo getInfo() { return info; } public OperatorStats add(OperatorStats... operators) { return add(ImmutableList.copyOf(operators)); } public OperatorStats add(Iterable<OperatorStats> operators) { long totalDrivers = this.totalDrivers; long addInputCalls = this.addInputCalls; long addInputWall = this.addInputWall.roundTo(NANOSECONDS); long addInputCpu = this.addInputCpu.roundTo(NANOSECONDS); long addInputUser = this.addInputUser.roundTo(NANOSECONDS); long inputDataSize = this.inputDataSize.toBytes(); long inputPositions = this.inputPositions; double sumSquaredInputPositions = this.sumSquaredInputPositions; long getOutputCalls = this.getOutputCalls; long getOutputWall = this.getOutputWall.roundTo(NANOSECONDS); long getOutputCpu = this.getOutputCpu.roundTo(NANOSECONDS); long getOutputUser = this.getOutputUser.roundTo(NANOSECONDS); long outputDataSize = this.outputDataSize.toBytes(); long outputPositions = this.outputPositions; long blockedWall = this.blockedWall.roundTo(NANOSECONDS); long finishCalls = this.finishCalls; long finishWall = this.finishWall.roundTo(NANOSECONDS); long finishCpu = this.finishCpu.roundTo(NANOSECONDS); long finishUser = this.finishUser.roundTo(NANOSECONDS); long memoryReservation = this.memoryReservation.toBytes(); long systemMemoryReservation = this.systemMemoryReservation.toBytes(); Optional<BlockedReason> blockedReason = this.blockedReason; Mergeable<OperatorInfo> base = getMergeableInfoOrNull(info); for (OperatorStats operator : operators) { checkArgument(operator.getOperatorId() == operatorId, "Expected operatorId to be %s but was %s", operatorId, operator.getOperatorId()); totalDrivers += operator.totalDrivers; addInputCalls += operator.getAddInputCalls(); addInputWall += operator.getAddInputWall().roundTo(NANOSECONDS); addInputCpu += operator.getAddInputCpu().roundTo(NANOSECONDS); addInputUser += operator.getAddInputUser().roundTo(NANOSECONDS); inputDataSize += operator.getInputDataSize().toBytes(); inputPositions += operator.getInputPositions(); sumSquaredInputPositions += operator.getSumSquaredInputPositions(); getOutputCalls += operator.getGetOutputCalls(); getOutputWall += operator.getGetOutputWall().roundTo(NANOSECONDS); getOutputCpu += operator.getGetOutputCpu().roundTo(NANOSECONDS); getOutputUser += operator.getGetOutputUser().roundTo(NANOSECONDS); outputDataSize += operator.getOutputDataSize().toBytes(); outputPositions += operator.getOutputPositions(); finishCalls += operator.getFinishCalls(); finishWall += operator.getFinishWall().roundTo(NANOSECONDS); finishCpu += operator.getFinishCpu().roundTo(NANOSECONDS); finishUser += operator.getFinishUser().roundTo(NANOSECONDS); blockedWall += operator.getBlockedWall().roundTo(NANOSECONDS); memoryReservation += operator.getMemoryReservation().toBytes(); systemMemoryReservation += operator.getSystemMemoryReservation().toBytes(); if (operator.getBlockedReason().isPresent()) { blockedReason = operator.getBlockedReason(); } OperatorInfo info = operator.getInfo(); if (base != null && info != null && base.getClass() == info.getClass()) { base = mergeInfo(base, info); } } return new OperatorStats( pipelineId, operatorId, planNodeId, operatorType, totalDrivers, addInputCalls, new Duration(addInputWall, NANOSECONDS).convertToMostSuccinctTimeUnit(), new Duration(addInputCpu, NANOSECONDS).convertToMostSuccinctTimeUnit(), new Duration(addInputUser, NANOSECONDS).convertToMostSuccinctTimeUnit(), succinctBytes(inputDataSize), inputPositions, sumSquaredInputPositions, getOutputCalls, new Duration(getOutputWall, NANOSECONDS).convertToMostSuccinctTimeUnit(), new Duration(getOutputCpu, NANOSECONDS).convertToMostSuccinctTimeUnit(), new Duration(getOutputUser, NANOSECONDS).convertToMostSuccinctTimeUnit(), succinctBytes(outputDataSize), outputPositions, new Duration(blockedWall, NANOSECONDS).convertToMostSuccinctTimeUnit(), finishCalls, new Duration(finishWall, NANOSECONDS).convertToMostSuccinctTimeUnit(), new Duration(finishCpu, NANOSECONDS).convertToMostSuccinctTimeUnit(), new Duration(finishUser, NANOSECONDS).convertToMostSuccinctTimeUnit(), succinctBytes(memoryReservation), succinctBytes(systemMemoryReservation), blockedReason, (OperatorInfo) base); } @SuppressWarnings("unchecked") private static Mergeable<OperatorInfo> getMergeableInfoOrNull(OperatorInfo info) { Mergeable<OperatorInfo> base = null; if (info instanceof Mergeable) { base = (Mergeable<OperatorInfo>) info; } return base; } @SuppressWarnings("unchecked") private static <T> Mergeable<T> mergeInfo(Mergeable<T> base, T other) { return (Mergeable<T>) base.mergeWith(other); } public OperatorStats summarize() { return new OperatorStats( pipelineId, operatorId, planNodeId, operatorType, totalDrivers, addInputCalls, addInputWall, addInputCpu, addInputUser, inputDataSize, inputPositions, sumSquaredInputPositions, getOutputCalls, getOutputWall, getOutputCpu, getOutputUser, outputDataSize, outputPositions, blockedWall, finishCalls, finishWall, finishCpu, finishUser, memoryReservation, systemMemoryReservation, blockedReason, (info != null && info.isFinal()) ? info : null); } }