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

org.apache.flink.runtime.taskmanager.TaskManagerLocation Maven / Gradle / Ivy

The 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 org.apache.flink.runtime.taskmanager;

import java.net.InetAddress;

import org.apache.flink.runtime.clusterframework.types.ResourceID;
import org.apache.flink.util.NetUtils;

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

import javax.annotation.Nonnull;

import static org.apache.flink.util.Preconditions.checkArgument;
import static org.apache.flink.util.Preconditions.checkNotNull;

/**
 * This class encapsulates the connection information of a TaskManager.
 * It describes the host where the TaskManager operates and its server port
 * for data exchange. This class also contains utilities to work with the
 * TaskManager's host name, which is used to localize work assignments.
 */
public class TaskManagerLocation implements Comparable, java.io.Serializable {

	private static final long serialVersionUID = -8254407801276350716L;

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

	// ------------------------------------------------------------------------

	/** The ID of the resource in which the TaskManager is started. This can be for example
	 * the YARN container ID, Mesos container ID, or any other unique identifier. */
	private final ResourceID resourceID;

	/** The network address that the TaskManager binds its sockets to */
	private final InetAddress inetAddress;

	/** The fully qualified host name of the TaskManager */
	private final String fqdnHostName;

	/** The pure hostname, derived from the fully qualified host name. */
	private final String hostName;
	
	/** The port that the TaskManager receive data transport connection requests at */
	private final int dataPort;

	/** The toString representation, eagerly constructed and cached to avoid repeated string building */  
	private final String stringRepresentation;

	/**
	 * Constructs a new instance connection info object. The constructor will attempt to retrieve the instance's
	 * host name and domain name through the operating system's lookup mechanisms.
	 * 
	 * @param inetAddress
	 *        the network address the instance's task manager binds its sockets to
	 * @param dataPort
	 *        the port instance's task manager expects to receive transfer envelopes on
	 */
	public TaskManagerLocation(ResourceID resourceID, InetAddress inetAddress, int dataPort) {
		// -1 indicates a local instance connection info
		checkArgument(dataPort > 0 || dataPort == -1, "dataPort must be > 0, or -1 (local)");

		this.resourceID = checkNotNull(resourceID);
		this.inetAddress = checkNotNull(inetAddress);
		this.dataPort = dataPort;

		// get FQDN hostname on this TaskManager.
		String fqdnHostName;
		try {
			fqdnHostName = this.inetAddress.getCanonicalHostName();
		}
		catch (Throwable t) {
			LOG.warn("Unable to determine the canonical hostname. Input split assignment (such as " +
					"for HDFS files) may be non-local when the canonical hostname is missing.");
			LOG.debug("getCanonicalHostName() Exception:", t);
			fqdnHostName = this.inetAddress.getHostAddress();
		}
		this.fqdnHostName = fqdnHostName;

		if (this.fqdnHostName.equals(this.inetAddress.getHostAddress())) {
			// this happens when the name lookup fails, either due to an exception,
			// or because no hostname can be found for the address
			// take IP textual representation
			this.hostName = this.fqdnHostName;
			LOG.warn("No hostname could be resolved for the IP address {}, using IP address as host name. "
					+ "Local input split assignment (such as for HDFS files) may be impacted.",
					this.inetAddress.getHostAddress());
		}
		else {
			this.hostName = NetUtils.getHostnameFromFQDN(this.fqdnHostName);
		}

		this.stringRepresentation = String.format(
				"%s @ %s (dataPort=%d)", resourceID, fqdnHostName, dataPort);
	}

	// ------------------------------------------------------------------------
	//  Getters
	// ------------------------------------------------------------------------

	/**
	 * Gets the ID of the resource in which the TaskManager is started. The format of this depends
	 * on how the TaskManager is started:
	 * 
    *
  • If the TaskManager is started via YARN, this is the YARN container ID.
  • *
  • If the TaskManager is started via Mesos, this is the Mesos container ID.
  • *
  • If the TaskManager is started in standalone mode, or via a MiniCluster, this is a random ID.
  • *
  • Other deployment modes can set the resource ID in other ways.
  • *
* * @return The ID of the resource in which the TaskManager is started */ public ResourceID getResourceID() { return resourceID; } /** * Returns the port instance's task manager expects to receive transfer envelopes on. * * @return the port instance's task manager expects to receive transfer envelopes on */ public int dataPort() { return dataPort; } /** * Returns the network address the instance's task manager binds its sockets to. * * @return the network address the instance's task manager binds its sockets to */ public InetAddress address() { return inetAddress; } /** * Gets the IP address where the TaskManager operates. * * @return The IP address. */ public String addressString() { return inetAddress.toString(); } /** * Returns the fully-qualified domain name the TaskManager. If the name could not be * determined, the return value will be a textual representation of the TaskManager's IP address. * * @return The fully-qualified domain name of the TaskManager. */ public String getFQDNHostname() { return fqdnHostName; } /** * Gets the hostname of the TaskManager. The hostname derives from the fully qualified * domain name (FQDN, see {@link #getFQDNHostname()}): *
    *
  • If the FQDN is the textual IP address, then the hostname is also the IP address
  • *
  • If the FQDN has only one segment (such as "localhost", or "host17"), then this is * used as the hostname.
  • *
  • If the FQDN has multiple segments (such as "worker3.subgroup.company.net"), then the first * segment (here "worker3") will be used as the hostname.
  • *
* * @return The hostname of the TaskManager. */ public String getHostname() { return hostName; } // -------------------------------------------------------------------------------------------- // Utilities // -------------------------------------------------------------------------------------------- @Override public String toString() { return stringRepresentation; } @Override public boolean equals(Object obj) { if (obj == this) { return true; } else if (obj != null && obj.getClass() == TaskManagerLocation.class) { TaskManagerLocation that = (TaskManagerLocation) obj; return this.resourceID.equals(that.resourceID) && this.inetAddress.equals(that.inetAddress) && this.dataPort == that.dataPort; } else { return false; } } @Override public int hashCode() { return resourceID.hashCode() + 17 * inetAddress.hashCode() + 129 * dataPort; } @Override public int compareTo(@Nonnull TaskManagerLocation o) { // decide based on resource ID first int resourceIdCmp = this.resourceID.getResourceIdString().compareTo(o.resourceID.getResourceIdString()); if (resourceIdCmp != 0) { return resourceIdCmp; } // decide based on ip address next byte[] thisAddress = this.inetAddress.getAddress(); byte[] otherAddress = o.inetAddress.getAddress(); if (thisAddress.length < otherAddress.length) { return -1; } else if (thisAddress.length > otherAddress.length) { return 1; } else { for (int i = 0; i < thisAddress.length; i++) { byte tb = thisAddress[i]; byte ob = otherAddress[i]; if (tb < ob) { return -1; } else if (tb > ob) { return 1; } } } // addresses are identical, decide based on ports. if (this.dataPort < o.dataPort) { return -1; } else if (this.dataPort > o.dataPort) { return 1; } else { return 0; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy