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

com.datatorrent.stram.BlacklistBasedResourceRequestHandler Maven / Gradle / Ivy

There is a newer version: 3.7.0
Show 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 com.datatorrent.stram;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import org.apache.commons.lang.StringUtils;
import org.apache.commons.lang3.tuple.MutablePair;
import org.apache.hadoop.yarn.client.api.AMRMClient;
import org.apache.hadoop.yarn.client.api.AMRMClient.ContainerRequest;

import com.datatorrent.stram.StreamingContainerAgent.ContainerStartRequest;

/**
 * Handles creating container requests and issuing node-specific container
 * requests by blacklisting (specifically for cloudera)
 * 
 * Host specific container requests are not allocated on Cloudera as captured in
 * Jira Yarn-1412 (https://issues.apache.org/jira/browse/YARN-1412) 
 * To handle such requests, we blacklist all the other nodes before issueing node request.
 *
 * @since 3.4.0
 */
public class BlacklistBasedResourceRequestHandler extends ResourceRequestHandler
{
  HashMap hostSpecificRequests = new HashMap<>();
  HashMap otherContainerRequests = new HashMap<>();
  HashMap> hostSpecificRequestsMap = new HashMap<>();
  List blacklistedNodesForHostSpecificRequests = null;

  public void reissueContainerRequests(AMRMClient amRmClient, Map> requestedResources, int loopCounter, ResourceRequestHandler resourceRequestor, List containerRequests, List removedContainerRequests)
  {
    // Issue all host specific requests first
    if (!hostSpecificRequestsMap.isEmpty() && requestedResources.isEmpty()) {
      LOG.info("Issue Host specific requests first");
      // Blacklist all the nodes and issue request for host specific
      Entry> set = hostSpecificRequestsMap.entrySet().iterator().next();
      List requests = set.getValue();
      List blacklistNodes = resourceRequestor.getNodesExceptHost(requests.get(0).getNodes());
      amRmClient.updateBlacklist(blacklistNodes, requests.get(0).getNodes());
      blacklistedNodesForHostSpecificRequests = blacklistNodes;
      LOG.info("Sending {} request(s) after blacklisting all nodes other than {}", requests.size(), requests.get(0).getNodes());

      for (ContainerRequest cr : requests) {
        ContainerStartRequest csr = hostSpecificRequests.get(cr);
        ContainerRequest newCr = new ContainerRequest(cr.getCapability(), null, null, cr.getPriority());
        MutablePair pair = new MutablePair(loopCounter, newCr);
        requestedResources.put(csr, pair);
        containerRequests.add(newCr);
        hostSpecificRequests.remove(cr);
      }
      hostSpecificRequestsMap.remove(set.getKey());
    } else if (!requestedResources.isEmpty()) {
      // Check if any requests timed out, create new requests in that case
      recreateContainerRequest(requestedResources, loopCounter, resourceRequestor, removedContainerRequests);
    } else {
      if (blacklistedNodesForHostSpecificRequests != null) {
        // Remove the blacklisted nodes during host specific requests
        LOG.debug("All requests done.. Removing nodes from blacklist {}", blacklistedNodesForHostSpecificRequests);
        amRmClient.updateBlacklist(null, blacklistedNodesForHostSpecificRequests);
        blacklistedNodesForHostSpecificRequests = null;
      }
      // Proceed with other requests after host specific requests are done
      if (!otherContainerRequests.isEmpty()) {
        for (Entry entry : otherContainerRequests.entrySet()) {
          ContainerRequest cr = entry.getKey();
          ContainerStartRequest csr = entry.getValue();
          MutablePair pair = new MutablePair(loopCounter, cr);
          requestedResources.put(csr, pair);
          containerRequests.add(cr);
        }
        otherContainerRequests.clear();
      }
    }
  }

  public void recreateContainerRequest(Map> requestedResources, int loopCounter, ResourceRequestHandler resourceRequestor, List removedContainerRequests)
  {
    for (Map.Entry> entry : requestedResources.entrySet()) {
      if ((loopCounter - entry.getValue().getKey()) > NUMBER_MISSED_HEARTBEATS) {
        StreamingContainerAgent.ContainerStartRequest csr = entry.getKey();
        removedContainerRequests.add(entry.getValue().getRight());
        ContainerRequest cr = resourceRequestor.createContainerRequest(csr, false);
        if (cr.getNodes() != null && !cr.getNodes().isEmpty()) {
          addHostSpecificRequest(csr, cr);
        } else {
          otherContainerRequests.put(cr, csr);
        }
      }
    }
  }

  public void addContainerRequest(Map> requestedResources, int loopCounter, List containerRequests, StreamingContainerAgent.ContainerStartRequest csr, ContainerRequest cr)
  {
    if (cr.getNodes() != null && !cr.getNodes().isEmpty()) {
      // Put it in a Map to check if multiple requests can be combined
      addHostSpecificRequest(csr, cr);
    } else {
      LOG.info("No node specific request ", cr);
      otherContainerRequests.put(cr, csr);
    }
  }

  public void addHostSpecificRequest(StreamingContainerAgent.ContainerStartRequest csr, ContainerRequest cr)
  {
    String hostKey = StringUtils.join(cr.getNodes(), ":");
    List requests;
    if (hostSpecificRequestsMap.containsKey(hostKey)) {
      requests = hostSpecificRequestsMap.get(hostKey);
    } else {
      requests = new ArrayList<>();
    }
    requests.add(cr);
    hostSpecificRequestsMap.put(hostKey, requests);
    LOG.info("Requesting container for node {} request = {}", cr.getNodes(), cr);
    hostSpecificRequests.put(cr, csr);
  }

  private static final Logger LOG = LoggerFactory.getLogger(BlacklistBasedResourceRequestHandler.class);
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy