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

org.apache.drill.yarn.client.StopCommand 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.yarn.client;

import org.apache.drill.yarn.core.DoYUtil;
import org.apache.drill.yarn.core.DrillOnYarnConfig;
import org.apache.drill.yarn.core.YarnClientException;
import org.apache.drill.yarn.core.YarnRMClient;
import org.apache.hadoop.yarn.api.records.YarnApplicationState;

import com.typesafe.config.Config;

/**
 * Perform a semi-graceful shutdown of the Drill-on-YARN AM. We send a message
 * to the AM to request shutdown because the YARN-provided message just kills
 * the AM. (There seems to be no way to get YARN to call its own
 * AMRMClientAsync.CallbackHandler.onShutdownRequest message.) The AM, however,
 * cannot gracefully shut down the drill-bits because Drill itself has no
 * graceful shutdown. But, at least this technique gives the AM a fighting
 * chance to do graceful shutdown in the future.
 */

public class StopCommand extends ClientCommand {
  /**
   * Poll the YARN RM to check the stop status of the AM. Periodically poll,
   * waiting to get an app state that indicates app completion.
   */

  private static class StopMonitor {
    StatusCommand.Reporter reporter;
    private YarnApplicationState state;
    private int pollWaitSec;
    private int shutdownWaitSec;

    StopMonitor(Config config, StatusCommand.Reporter reporter) {
      this.reporter = reporter;
      pollWaitSec = config.getInt(DrillOnYarnConfig.CLIENT_POLL_SEC);
      if (pollWaitSec < 1) {
        pollWaitSec = 1;
      }
      shutdownWaitSec = config.getInt(DrillOnYarnConfig.CLIENT_STOP_WAIT_SEC);
    }

    boolean run(boolean verbose) throws ClientException {
      System.out.print("Stopping...");
      try {
        int attemptCount = shutdownWaitSec / pollWaitSec;
        for (int attempt = 0; attempt < attemptCount; attempt++) {
          if (!poll()) {
            break;
          }
        }
      } catch (ClientException e) {
        System.out.println();
        throw e;
      }
      if (reporter.isStopped()) {
        System.out.println(" Stopped.");
        reporter.showFinalStatus();
        return true;
      } else {
        System.out.println();
        System.out.println(
            "Application Master is slow to stop, use YARN to check status.");
        return false;
      }
    }

    private boolean poll() throws ClientException {
      try {
        Thread.sleep(pollWaitSec * 1000);
      } catch (InterruptedException e) {
        return false;
      }
      reporter.getReport();
      if (reporter.isStopped()) {
        return false;
      }
      YarnApplicationState newState = reporter.getState();
      if (newState == state) {
        System.out.print(".");
        return true;
      }
      updateState(newState);
      return true;
    }

    private void updateState(YarnApplicationState newState) {
      YarnApplicationState oldState = state;
      state = newState;
      if (oldState == null) {
        return;
      }
      System.out.println();
      System.out.print("Application State: ");
      System.out.println(state.toString());
      System.out.print("Stopping...");
    }
  }

  private Config config;
  private YarnRMClient client;

  @Override
  public void run() throws ClientException {
    config = DrillOnYarnConfig.config();
    client = getClient();
    System.out
        .println("Stopping Application ID: " + client.getAppId().toString());

    // First get an application report to ensure that the AM is,
    // in fact, running, and to get the HTTP endpoint.

    StatusCommand.Reporter reporter = new StatusCommand.Reporter(client);
    try {
      reporter.getReport();
    } catch (ClientException e) {
      reporter = null;
    }

    // Handle the case of an already stopped app.

    boolean stopped = true;
    if (reporter == null || reporter.isStopped()) {
      System.out.println("Application is not running.");
    } else {
      // Try to stop the server by sending a STOP REST request.

      if (opts.force) {
        System.out.println("Forcing shutdown");
      } else {
        stopped = gracefulStop(reporter.getAmUrl());
      }

      // If that did not work, then forcibly kill the AM.
      // YARN will forcibly kill the AM's containers.
      // Not pretty, but it works.

      if (opts.force || !stopped) {
        forcefulStop();
      }

      // Wait for the AM to stop. The AM may refuse to stop in
      // the time allowed to wait.

      stopped = new StopMonitor(config, reporter).run(opts.verbose);
    }

    // If the AM is gone because it started out dead or
    // we killed it, then forget its App Id.

    if (stopped) {
      removeAppIdFile();
    }
  }

  /**
   * Do a graceful shutdown by using the AM's REST API call to request stop.
   * Include the master key with the request to differentiate this request from
   * accidental uses of the stop REST API.
   *
   * @param report
   * @return
   */

  private boolean gracefulStop(String baseUrl) {
    try {
      if (DoYUtil.isBlank(baseUrl)) {
        return false;
      }
      SimpleRestClient restClient = new SimpleRestClient();
      String tail = "rest/stop";
      String masterKey = config.getString(DrillOnYarnConfig.HTTP_REST_KEY);
      if (!DoYUtil.isBlank(masterKey)) {
        tail += "?key=" + masterKey;
      }
      if (opts.verbose) {
        System.out.println("Stopping with POST " + baseUrl + "/" + tail);
      }
      String result = restClient.send(baseUrl, tail, true);
      if (result.contains("\"ok\"")) {
        return true;
      }
      System.err.println(
          "Failed to stop the application master. Response = " + result);
      return false;
    } catch (ClientException e) {
      System.err.println(e.getMessage());
      System.out.println("Resorting to forced kill");
      return false;
    }
  }

  /**
   * If the graceful approach did not work, resort to a forceful request. This
   * asks the AM's NM to kill the AM process.
   *
   * @throws ClientException
   */

  private void forcefulStop() throws ClientException {
    try {
      client.killApplication();
    } catch (YarnClientException e) {
      throw new ClientException("Failed to stop application master", e);
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy