io.micronaut.discovery.eureka.client.v2.InstanceInfo Maven / Gradle / Ivy
/*
* Copyright 2017-2020 original authors
*
* 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
*
* https://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 io.micronaut.discovery.eureka.client.v2;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.annotation.JsonRootName;
import io.micronaut.core.util.StringUtils;
import io.micronaut.serde.annotation.Serdeable;
import jakarta.validation.constraints.Min;
import jakarta.validation.constraints.NotBlank;
import jakarta.validation.constraints.NotNull;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
/**
* Represents an application instance in Eureka. See https://github.com/Netflix/eureka/wiki/Eureka-REST-operations.
*
* Based on https://github.com/Netflix/eureka/blob/master/eureka-client/src/main/java/com/netflix/appinfo/InstanceInfo.java
*
* @author graemerocher
* @since 1.0
*/
@JsonRootName("instance")
@Serdeable
public class InstanceInfo implements ConfigurableInstanceInfo {
/**
* Eureka default port.
*/
public static final int DEFAULT_PORT = 7001;
/**
* Secure port disabled by default.
*/
public static final int DEFAULT_SECURE_PORT = 0;
/**
* US by default.
*/
public static final int DEFAULT_COUNTRY_ID = 1;
private String asgName;
private String hostName;
private String app;
private String instanceId;
private String ipAddr;
private String appGroupName;
private String vipAddress;
private String secureVipAddress;
private Status status = Status.UP;
private int port = DEFAULT_PORT;
private int securePort = DEFAULT_SECURE_PORT;
private int countryId = DEFAULT_COUNTRY_ID; // Defaults to US
private String homePageUrl;
private String statusPageUrl;
private String healthCheckUrl;
private String secureHealthCheckUrl;
private DataCenterInfo dataCenterInfo = () -> DataCenterInfo.Name.MyOwn;
private LeaseInfo leaseInfo;
private Map metadata = new ConcurrentHashMap<>();
/**
* Based on https://github.com/Netflix/eureka/blob/master/eureka-client/src/main/java/com/netflix/appinfo/InstanceInfo.java.
*
* @param instanceId The instance id
* @param appName The application name
* @param appGroupName The application group name
* @param ipAddr The IP address
* @param port The port
* @param securePort The secure port
* @param homePageUrl The homepage URL
* @param statusPageUrl The status page URL
* @param healthCheckUrl The health check URL
* @param secureHealthCheckUrl The secure health check URL
* @param vipAddress The VIP address
* @param secureVipAddress The secure VIP address
* @param countryId The country ID
* @param dataCenterInfo The data center info
* @param hostName The hostname
* @param status The status
* @param overriddenstatus The overridden status
* @param leaseInfo The lease info
* @param metadata The metadata
* @param asgName The asg name
*/
@SuppressWarnings("ParameterNumber")
@JsonCreator
InstanceInfo(
@JsonProperty("instanceId") String instanceId,
@JsonProperty("app") String appName,
@JsonProperty("appGroupName") String appGroupName,
@JsonProperty("ipAddr") String ipAddr,
@JsonProperty("port") PortWrapper port,
@JsonProperty("securePort") PortWrapper securePort,
@JsonProperty("homePageUrl") String homePageUrl,
@JsonProperty("statusPageUrl") String statusPageUrl,
@JsonProperty("healthCheckUrl") String healthCheckUrl,
@JsonProperty("secureHealthCheckUrl") String secureHealthCheckUrl,
@JsonProperty("vipAddress") String vipAddress,
@JsonProperty("secureVipAddress") String secureVipAddress,
@JsonProperty("countryId") int countryId,
@JsonProperty("dataCenterInfo") DataCenterInfo dataCenterInfo,
@JsonProperty("hostName") String hostName,
@JsonProperty("status") Status status,
@JsonProperty("overriddenstatus") Status overriddenstatus,
@JsonProperty("leaseInfo") LeaseInfo leaseInfo,
@JsonProperty("metadata") HashMap metadata,
@JsonProperty("asgName") String asgName) {
this.instanceId = instanceId;
this.app = appName;
this.appGroupName = appGroupName;
this.ipAddr = ipAddr;
this.port = port == null ? 0 : port.getPort();
this.securePort = securePort == null ? 0 : securePort.getPort();
this.homePageUrl = homePageUrl;
this.statusPageUrl = statusPageUrl;
this.healthCheckUrl = healthCheckUrl;
this.secureHealthCheckUrl = secureHealthCheckUrl;
this.vipAddress = vipAddress;
this.secureVipAddress = secureVipAddress;
this.countryId = countryId;
this.dataCenterInfo = dataCenterInfo;
this.hostName = hostName;
this.status = status;
this.leaseInfo = leaseInfo;
this.asgName = asgName;
// ---------------------------------------------------------------
// for compatibility
this.metadata = Objects.requireNonNullElse(metadata, Collections.emptyMap());
}
/**
* Creates an {@link InstanceInfo}.
*
* @param host The host name
* @param appName The application name
* @param instanceId The instance identifier
*/
public InstanceInfo(String host, @NotBlank String appName, @NotBlank String instanceId) {
this(host, DEFAULT_PORT, appName, instanceId);
}
/**
* Creates an {@link InstanceInfo}. The {@link #getInstanceId()} will default to the value of the host
*
* @param host The host name
* @param appName The application name
*/
public InstanceInfo(String host, @NotBlank String appName) {
this(host, DEFAULT_PORT, appName, host);
}
/**
* Creates an {@link InstanceInfo}. This constructor will perform an IP Address lookup based on the host name
*
* @param host The host name
* @param port The port
* @param appName The application name
* @param instanceId The instance identifier
*/
public InstanceInfo(String host, int port, @NotBlank String appName, @NotBlank String instanceId) {
this(host, port, lookupIp(host), appName, instanceId);
}
/**
* Creates an {@link InstanceInfo}.
*
* @param host The host name
* @param port The port
* @param ipAddress The IP address
* @param appName The application name
* @param instanceId The instance identifier
*/
public InstanceInfo(String host, int port, String ipAddress, String appName, String instanceId) {
if (StringUtils.isEmpty(host)) {
throw new IllegalArgumentException("Argument [hostName] cannot be null or blank");
}
if (StringUtils.isEmpty(appName)) {
throw new IllegalArgumentException("Argument [appName] cannot be null or blank");
}
if (StringUtils.isEmpty(instanceId)) {
throw new IllegalArgumentException("Argument [instanceId] cannot be null or blank");
}
if (StringUtils.isEmpty(ipAddress)) {
throw new IllegalArgumentException("Argument [ipAddress] cannot be null or blank");
}
this.hostName = host;
this.port = port;
this.ipAddr = ipAddress;
this.app = appName;
this.instanceId = instanceId;
}
@Override
public String toString() {
return getId();
}
/**
* The host name of the application instance.
*/
@Override
@NotBlank
public String getHostName() {
return hostName;
}
/**
* Returns the unique id of the instance.
* (Note) now that id is set at creation time within the instanceProvider, why do the other checks?
* This is still necessary for backwards compatibility when upgrading in a deployment with multiple
* client versions (some with the change, some without).
*
* @return the unique id.
*/
@Override
@JsonIgnore
public String getId() {
if (instanceId != null && !instanceId.isEmpty()) {
return instanceId;
} else if (dataCenterInfo instanceof AmazonInfo amazonInfo) {
String uniqueId = amazonInfo.getId();
if (uniqueId != null && !uniqueId.isEmpty()) {
return uniqueId;
}
}
return hostName;
}
/**
* The port of the application instance.
*/
@Override
@JsonIgnore
public int getPort() {
return port;
}
/**
* The secure port of the application instance.
*/
@Override
@JsonIgnore
public int getSecurePort() {
return securePort;
}
/**
* @return The port
*/
@JsonProperty("port")
public PortWrapper getPortWrapper() {
if (port < 1) {
return new PortWrapper(false, 0);
}
return new PortWrapper(true, port);
}
/**
* @return The secure port
*/
@JsonProperty("securePort")
public PortWrapper getSecurePortWrapper() {
if (securePort < 1) {
return new PortWrapper(false, 0);
}
return new PortWrapper(true, securePort);
}
@Override
public void setSecurePort(int securePort) {
this.securePort = securePort;
}
@Override
public void setPort(int port) {
if (port >= 0) {
this.port = port;
}
}
/**
* The application name.
*/
@Override
@NotBlank
public String getApp() {
return app;
}
/**
* The application group name.
*/
@Override
public String getAppGroupName() {
return appGroupName;
}
/**
* The instance id.
*/
@Override
@NotBlank
public String getInstanceId() {
return instanceId;
}
/**
* The country id.
*/
@Override
@Min(1L)
public int getCountryId() {
return countryId;
}
/**
* The IP address of the instance.
*/
@Override
@NotBlank
public String getIpAddr() {
return ipAddr;
}
@Override
@NotNull
public Status getStatus() {
return status;
}
/**
* The {@link DataCenterInfo} instance.
*/
@Override
@NotNull
public DataCenterInfo getDataCenterInfo() {
return dataCenterInfo;
}
/**
* The {@link LeaseInfo} instance.
*/
@Override
public LeaseInfo getLeaseInfo() {
return leaseInfo;
}
/**
* @return The instance metadata.
*/
@Override
public Map getMetadata() {
return metadata;
}
/**
* @return The status page URL
*/
@Override
public String getStatusPageUrl() {
if (this.statusPageUrl == null) {
return getHealthCheckUrl();
}
return statusPageUrl;
}
/**
* @return The home page URL
*/
@Override
public String getHomePageUrl() {
if (this.homePageUrl == null) {
return "http://" + this.hostName + portString();
}
return homePageUrl;
}
/**
* @return The health check URL
*/
@Override
public String getHealthCheckUrl() {
return Objects.requireNonNullElseGet(this.healthCheckUrl, () -> "http://" + this.hostName + portString() + "/health");
}
/**
* @return The Virtual Host Address for this instance (defaults to the app name)
*/
@Override
@NotBlank
public String getVipAddress() {
if (this.vipAddress == null) {
return this.app;
}
return vipAddress;
}
/**
* @return The Secure Virtual Host Address for this instance (defaults to the app name)
*/
@Override
@NotBlank
public String getSecureVipAddress() {
if (this.secureVipAddress == null) {
return this.app;
}
return secureVipAddress;
}
/**
* @return The secure health check URL
*/
@Override
public String getSecureHealthCheckUrl() {
return Objects.requireNonNullElseGet(this.secureHealthCheckUrl, () -> "https://" + this.hostName + securePortString() + "/health");
}
/**
* @return The amazon auto scaling group name
*/
@Override
public String getAsgName() {
return asgName;
}
/**
* Sets the instance ID.
*
* @param instanceId The instance ID
*/
@Override
public void setInstanceId(String instanceId) {
this.instanceId = instanceId;
}
@Override
public void setAsgName(String asgName) {
this.asgName = asgName;
}
@Override
public void setHomePageUrl(String homePageUrl) {
if (!StringUtils.isEmpty(homePageUrl)) {
this.homePageUrl = homePageUrl;
}
}
@Override
public void setLeaseInfo(LeaseInfo leaseInfo) {
this.leaseInfo = leaseInfo;
}
@Override
public void setCountryId(int countryId) {
if (countryId > 0) {
this.countryId = countryId;
}
}
@Override
public void setStatusPageUrl(String statusPageUrl) {
if (!StringUtils.isEmpty(statusPageUrl)) {
this.statusPageUrl = statusPageUrl;
}
}
@Override
public void setHealthCheckUrl(String healthCheckUrl) {
if (!StringUtils.isEmpty(healthCheckUrl)) {
this.healthCheckUrl = healthCheckUrl;
}
}
@Override
public void setSecureHealthCheckUrl(String secureHealthCheckUrl) {
if (!StringUtils.isEmpty(secureHealthCheckUrl)) {
this.secureHealthCheckUrl = secureHealthCheckUrl;
}
}
@Override
public void setDataCenterInfo(DataCenterInfo dataCenterInfo) {
if (dataCenterInfo != null) {
this.dataCenterInfo = dataCenterInfo;
}
}
@Override
public void setStatus(Status status) {
if (status != null) {
this.status = status;
}
}
@Override
public void setAppGroupName(String appGroupName) {
if (StringUtils.isNotEmpty(appGroupName)) {
this.appGroupName = appGroupName;
}
}
@Override
public void setIpAddr(String ipAddr) {
if (StringUtils.isNotEmpty(ipAddr)) {
this.ipAddr = ipAddr;
}
}
@Override
public void setVipAddress(String vipAddress) {
if (StringUtils.isNotEmpty(vipAddress)) {
this.vipAddress = vipAddress;
}
}
@Override
public void setSecureVipAddress(String secureVipAddress) {
if (StringUtils.isNotEmpty(secureVipAddress)) {
this.secureVipAddress = secureVipAddress;
}
}
/**
* @param metadata Sets the application metadata
*/
public void setMetadata(Map metadata) {
if (metadata != null) {
this.metadata = metadata;
}
}
private String portString() {
return port > 0 ? ":" + this.port : "";
}
private String securePortString() {
return securePort > 0 ? ":" + this.securePort : "";
}
private static String lookupIp(String host) {
try {
return InetAddress.getByName(host).getHostAddress();
} catch (UnknownHostException e) {
throw new IllegalArgumentException("Unable to lookup host IP address: " + host, e);
}
}
/**
* The instance status according to Eureka.
*/
public enum Status {
UP, DOWN, STARTING, OUT_OF_SERVICE, UNKNOWN
}
/**
* {@link InstanceInfo} JSON and XML format for port information does not follow the usual conventions, which
* makes its mapping complicated. This class represents the wire format for port information.
*/
static class PortWrapper {
private final boolean enabled;
private final int port;
/**
* @param enabled Whether is enabled
* @param port The port
*/
@JsonCreator
public PortWrapper(@JsonProperty("@enabled") boolean enabled, @JsonProperty("$") int port) {
this.enabled = enabled;
this.port = port;
}
/**
* @return Whether is enabled
*/
@JsonProperty("@enabled")
public boolean isEnabled() {
return enabled;
}
/**
* @return The port
*/
@JsonProperty("$")
public int getPort() {
return port;
}
}
}