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

org.apache.drill.exec.physical.impl.BaseRootExec Maven / Gradle / Ivy

/*
 * 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.drill.exec.physical.impl;

import java.util.List;

import org.apache.drill.common.DeferredException;
import org.apache.drill.exec.exception.OutOfMemoryException;
import org.apache.drill.exec.ops.OpProfileDef;
import org.apache.drill.exec.ops.OperatorContext;
import org.apache.drill.exec.ops.OperatorStats;
import org.apache.drill.exec.ops.OperatorUtilities;
import org.apache.drill.exec.ops.RootFragmentContext;
import org.apache.drill.exec.physical.base.PhysicalOperator;
import org.apache.drill.exec.proto.ExecProtos.FragmentHandle;
import org.apache.drill.exec.record.CloseableRecordBatch;
import org.apache.drill.exec.record.RecordBatch;
import org.apache.drill.exec.record.RecordBatch.IterOutcome;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public abstract class BaseRootExec implements RootExec {
  private static final Logger logger = LoggerFactory.getLogger(BaseRootExec.class);

  public static final String ENABLE_BATCH_DUMP_CONFIG = "drill.exec.debug.dump_batches";
  protected OperatorStats stats;
  protected OperatorContext oContext;
  protected RootFragmentContext fragmentContext;
  private List operators;

  public BaseRootExec(final RootFragmentContext fragmentContext, final PhysicalOperator config) throws OutOfMemoryException {
    this(fragmentContext, null, config);
  }

  public BaseRootExec(final RootFragmentContext fragmentContext, final OperatorContext oContext,
                      final PhysicalOperator config) throws OutOfMemoryException {
    if (oContext == null) {
      this.oContext = fragmentContext.newOperatorContext(config, stats);
    } else {
      this.oContext = oContext;
    }
    //Creating new stat for appending to list
    stats = new OperatorStats(new OpProfileDef(config.getOperatorId(),
        config.getOperatorType(), OperatorUtilities.getChildCount(config)),
      this.oContext.getAllocator());
    fragmentContext.getStats().addOperatorStats(this.stats);
    this.fragmentContext = fragmentContext;
  }

  void setOperators(List operators) {
    this.operators = operators;

    if (logger.isDebugEnabled()) {
      final StringBuilder sb = new StringBuilder();
      sb.append("BaseRootExec(");
      sb.append(Integer.toString(System.identityHashCode(this)));
      sb.append(") operators: ");
      for(final CloseableRecordBatch crb : operators) {
        sb.append(crb.getClass().getName());
        sb.append(' ');
        sb.append(Integer.toString(System.identityHashCode(crb)));
        sb.append(", ");
      }

      // Cut off the last trailing comma and space
      sb.setLength(sb.length() - 2);

      logger.debug(sb.toString());
    }
  }

  @Override
  public final boolean next() {
    assert stats != null;
    fragmentContext.getExecutorState().checkContinue();
    try {
      stats.startProcessing();
      return innerNext();
    } finally {
      stats.stopProcessing();
    }
  }

  public final IterOutcome next(final RecordBatch b){
    stats.stopProcessing();
    IterOutcome next;
    try {
      next = b.next();
    } finally {
      stats.startProcessing();
    }

    switch(next){
      case OK_NEW_SCHEMA:
        stats.batchReceived(0, b.getRecordCount(), true);
        break;
      case OK:
        stats.batchReceived(0, b.getRecordCount(), false);
        break;
      default:
    }
    return next;
  }

  public abstract boolean innerNext();

  @Override
  public void receivingFragmentFinished(final FragmentHandle handle) {
    logger.warn("Currently not handling FinishedFragment message");
  }

  @Override
  public void dumpBatches(Throwable t) {
    if (operators == null) {
      return;
    }
    if (!fragmentContext.getConfig().getBoolean(ENABLE_BATCH_DUMP_CONFIG)) {
      return;
    }

    CloseableRecordBatch leafMost = findLeaf(operators, t);
    if (leafMost == null) {
      // Don't know which batch failed.
      return;
    }
    int batchPosn = operators.indexOf(leafMost);
    final int numberOfBatchesToDump = Math.min(batchPosn + 1, 2);
    logger.error("Batch dump started: dumping last {} failed batches", numberOfBatchesToDump);
    // As batches are stored in a 'flat' List there is a need to filter out the failed batch
    // and a few of its parent (actual number of batches is set by a constant defined above)
    for (int i = 0; i < numberOfBatchesToDump; i++) {
      operators.get(batchPosn--).dump();
    }
    logger.error("Batch dump completed.");
  }

  @Override
  public void close() throws Exception {
    // We want to account for the time spent waiting here as Wait time in the operator profile
    try {
      stats.startProcessing();
      stats.startWait();
      fragmentContext.waitForSendComplete();
    } finally {
      stats.stopWait();
      stats.stopProcessing();
    }

    // close all operators.
    if (operators != null) {
      final DeferredException df = new DeferredException();

      for (final CloseableRecordBatch crb : operators) {
        df.suppressingClose(crb);
        if (logger.isDebugEnabled()) {
          logger.debug(String.format("closed operator %d", System.identityHashCode(crb)));
        }
      }

      try {
        df.close();
      } catch (Exception e) {
        fragmentContext.getExecutorState().fail(e);
      }
    }
  }

  /**
   * Given a list of operators and a stack trace, walks the stack trace and
   * the operator list to find the leaf-most operator, which is the one
   * that was active when the exception was thrown. Handle the cases in
   * which no operator was active, each operator had multiple methods on
   * the stack, or the exception was thrown in some class called by
   * the operator.
   * 

* Not all operators leave a mark in the trace. In particular if a the * call stack is only through base-class methods, then we have no way to * know the actual class during the call. This is OK because the leaf * methods are just pass-through operations, they are unlikely to fail. * * @param the type of the operator. Parameterized to allow easier * testing * @param dag the list of operators from root-most to leaf-most * @param e the exception thrown somewhere in the operator tree * @return the leaf-most operator, if any */ public static T findLeaf(List dag, Throwable e) { StackTraceElement[] trace = e.getStackTrace(); for (int i = dag.size() - 1; i >= 0; i--) { T leaf = dag.get(i); String opName = leaf.getClass().getName(); for (StackTraceElement element : trace) { String frameName = element.getClassName(); if (frameName.contentEquals(opName)) { return leaf; } } } return null; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy