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

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; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy