Spring Cloud Commons 提供的抽象java
最先的時候服務發現註冊都是經過DiscoveryClient來實現的,隨着版本變遷把DiscoveryClient服務註冊抽離出來變成了ServiceRegistry抽象,專門負責服務註冊,DiscoveryClient專門負責服務發現。還提供了負載均衡的發現LoadBalancerClient抽象。spring
DiscoveryClient經過@EnableDiscoveryClient的方式進行啓用。apache
自動向Eureka服務端註冊安全
ServiceRegistry使用的式EurekaServiceRegistry 來作的註冊, 註冊信息放在EurekaRegistration中app
源碼:負載均衡
public interface ServiceRegistry<R extends Registration> { /**註冊 */ void register(R registration); /** * 取消註冊 * */ void deregister(R registration); /** * 關閉服務 */ void close(); /** *設置狀態 */ void setStatus(R registration, String status); /** * 獲取狀態 */ <T> T getStatus(R registration); }
EurekaServiceRegistry的實現ide
import java.util.HashMap; import com.netflix.appinfo.InstanceInfo; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.springframework.cloud.client.serviceregistry.ServiceRegistry; import static com.netflix.appinfo.InstanceInfo.InstanceStatus.UNKNOWN; /** * @author Spencer Gibb */ public class EurekaServiceRegistry implements ServiceRegistry<EurekaRegistration> { private static final Log log = LogFactory.getLog(EurekaServiceRegistry.class);
// 獲取到EurekaRegistration 信息
@Override public void register(EurekaRegistration reg) {
// 初始化信息 maybeInitializeClient(reg); if (log.isInfoEnabled()) { log.info("Registering application " + reg.getApplicationInfoManager().getInfo().getAppName() + " with eureka with status " + reg.getInstanceConfig().getInitialStatus()); } // 設置實例狀態 reg.getApplicationInfoManager() .setInstanceStatus(reg.getInstanceConfig().getInitialStatus()); // 若是有healthCheckHandler 設置 healthCheck reg.getHealthCheckHandler().ifAvailable(healthCheckHandler -> reg .getEurekaClient().registerHealthCheck(healthCheckHandler)); } private void maybeInitializeClient(EurekaRegistration reg) { // force initialization of possibly scoped proxies reg.getApplicationInfoManager().getInfo(); reg.getEurekaClient().getApplications(); } @Override public void deregister(EurekaRegistration reg) { if (reg.getApplicationInfoManager().getInfo() != null) { if (log.isInfoEnabled()) { log.info("Unregistering application " + reg.getApplicationInfoManager().getInfo().getAppName() + " with eureka with status DOWN"); } //將狀態設置爲DOWN reg.getApplicationInfoManager() .setInstanceStatus(InstanceInfo.InstanceStatus.DOWN); // shutdown of eureka client should happen with EurekaRegistration.close() // auto registration will create a bean which will be properly disposed // manual registrations will need to call close() } } @Override public void setStatus(EurekaRegistration registration, String status) { InstanceInfo info = registration.getApplicationInfoManager().getInfo(); // TODO: howto deal with delete properly? if ("CANCEL_OVERRIDE".equalsIgnoreCase(status)) { registration.getEurekaClient().cancelOverrideStatus(info); return; } // TODO: howto deal with status types across discovery systems? InstanceInfo.InstanceStatus newStatus = InstanceInfo.InstanceStatus .toEnum(status); registration.getEurekaClient().setStatus(newStatus, info); } @Override public Object getStatus(EurekaRegistration registration) { String appname = registration.getApplicationInfoManager().getInfo().getAppName(); String instanceId = registration.getApplicationInfoManager().getInfo().getId(); InstanceInfo info = registration.getEurekaClient().getInstanceInfo(appname, instanceId); HashMap<String, Object> status = new HashMap<>(); if (info != null) { status.put("status", info.getStatus().toString()); status.put("overriddenStatus", info.getOverriddenStatus().toString()); } else { status.put("status", UNKNOWN.toString()); } return status; } public void close() { } }
DefaultLifecycleProcessor中start的調用
public void start() { startBeans(false); this.running = true; } private void startBeans(boolean autoStartupOnly) {
// 找到全部聲明週期中的bean Map<String, Lifecycle> lifecycleBeans = getLifecycleBeans(); Map<Integer, LifecycleGroup> phases = new HashMap<>(); lifecycleBeans.forEach((beanName, bean) -> { if (!autoStartupOnly || (bean instanceof SmartLifecycle && ((SmartLifecycle) bean).isAutoStartup())) { int phase = getPhase(bean); LifecycleGroup group = phases.get(phase); if (group == null) { group = new LifecycleGroup(phase, this.timeoutPerShutdownPhase, lifecycleBeans, autoStartupOnly); phases.put(phase, group); } group.add(beanName, bean); } }); if (!phases.isEmpty()) { List<Integer> keys = new ArrayList<>(phases.keySet()); Collections.sort(keys); for (Integer key : keys) {
// 只要不爲空,每個調用start方法 phases.get(key).start(); } } }
public void start() { if (this.members.isEmpty()) { return; } if (logger.isDebugEnabled()) { logger.debug("Starting beans in phase " + this.phase); } Collections.sort(this.members); for (LifecycleGroupMember member : this.members) { doStart(this.lifecycleBeans, member.name, this.autoStartupOnly); } }
private void doStart(Map<String, ? extends Lifecycle> lifecycleBeans, String beanName, boolean autoStartupOnly) { Lifecycle bean = lifecycleBeans.remove(beanName); if (bean != null && bean != this) { String[] dependenciesForBean = getBeanFactory().getDependenciesForBean(beanName); for (String dependency : dependenciesForBean) { doStart(lifecycleBeans, dependency, autoStartupOnly); } if (!bean.isRunning() && (!autoStartupOnly || !(bean instanceof SmartLifecycle) || ((SmartLifecycle) bean).isAutoStartup())) { if (logger.isTraceEnabled()) { logger.trace("Starting bean '" + beanName + "' of type [" + bean.getClass().getName() + "]"); } try { bean.start(); } catch (Throwable ex) { throw new ApplicationContextException("Failed to start bean '" + beanName + "'", ex); } if (logger.isDebugEnabled()) { logger.debug("Successfully started bean '" + beanName + "'"); } } } }
DiscoveryClientui
public interface DiscoveryClient extends Ordered { /** * Default order of the discovery client. */ int DEFAULT_ORDER = 0; /** * A human-readable description of the implementation, used in HealthIndicator. 提供一些描述信息 * @return The description. */ String description(); /** * Gets all ServiceInstances associated with a particular serviceId. * @param serviceId The serviceId to query. * @return A List of ServiceInstance. 根據serviceid 找到ServiceInstance */ List<ServiceInstance> getInstances(String serviceId); /** * @return All known service IDs. 獲取全部的服務 */ List<String> getServices(); /** * Default implementation for getting order of discovery clients. * @return order */ @Override default int getOrder() { return DEFAULT_ORDER; } }
Eureka的實現this
import java.net.URI; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.Map; import com.netflix.appinfo.EurekaInstanceConfig; import com.netflix.appinfo.InstanceInfo; import com.netflix.discovery.EurekaClient; import com.netflix.discovery.EurekaClientConfig; import com.netflix.discovery.shared.Application; import com.netflix.discovery.shared.Applications; import org.springframework.cloud.client.DefaultServiceInstance; import org.springframework.cloud.client.ServiceInstance; import org.springframework.cloud.client.discovery.DiscoveryClient; import org.springframework.core.Ordered; import org.springframework.util.Assert; import static com.netflix.appinfo.InstanceInfo.PortType.SECURE; /** * A {@link DiscoveryClient} implementation for Eureka. * * @author Spencer Gibb * @author Tim Ysewyn */ public class EurekaDiscoveryClient implements DiscoveryClient { /** * Client description {@link String}. */ public static final String DESCRIPTION = "Spring Cloud Eureka Discovery Client"; private final EurekaClient eurekaClient; private final EurekaClientConfig clientConfig; @Deprecated public EurekaDiscoveryClient(EurekaInstanceConfig config, EurekaClient eurekaClient) { this(eurekaClient, eurekaClient.getEurekaClientConfig()); } public EurekaDiscoveryClient(EurekaClient eurekaClient, EurekaClientConfig clientConfig) { this.clientConfig = clientConfig; this.eurekaClient = eurekaClient; } @Override public String description() { return DESCRIPTION; } // 經過serviceid 獲取到 instanceInfo信息 @Override public List<ServiceInstance> getInstances(String serviceId) { List<InstanceInfo> infos = this.eurekaClient.getInstancesByVipAddress(serviceId, false); List<ServiceInstance> instances = new ArrayList<>(); for (InstanceInfo info : infos) { instances.add(new EurekaServiceInstance(info)); } return instances; } @Override public List<String> getServices() { Applications applications = this.eurekaClient.getApplications(); if (applications == null) { return Collections.emptyList(); } List<Application> registered = applications.getRegisteredApplications(); List<String> names = new ArrayList<>(); for (Application app : registered) { if (app.getInstances().isEmpty()) { continue; } names.add(app.getName().toLowerCase()); } return names; } @Override public int getOrder() { return clientConfig instanceof Ordered ? ((Ordered) clientConfig).getOrder() : DiscoveryClient.DEFAULT_ORDER; } /** * An Eureka-specific {@link ServiceInstance} implementation. */ public static class EurekaServiceInstance implements ServiceInstance { private InstanceInfo instance; public EurekaServiceInstance(InstanceInfo instance) { Assert.notNull(instance, "Service instance required"); this.instance = instance; } public InstanceInfo getInstanceInfo() { return instance; } @Override public String getInstanceId() { return this.instance.getId(); } @Override public String getServiceId() { return this.instance.getAppName(); } @Override public String getHost() { return this.instance.getHostName(); } @Override public int getPort() { if (isSecure()) { return this.instance.getSecurePort(); } return this.instance.getPort(); } @Override public boolean isSecure() { // assume if secure is enabled, that is the default return this.instance.isPortEnabled(SECURE); } @Override public URI getUri() { return DefaultServiceInstance.getUri(this); } @Override public Map<String, String> getMetadata() { return this.instance.getMetadata(); } } }
總結:
自動配置註冊:
EurekaAutoServiceRegistration::start(在DefaultLifecycleProcessor中調用):
1. 先檢查port;
2. 若是沒有啓動過而且非安全的端口大於0,則進行註冊(經過調用org.springframework.cloud.client.serviceregistry.ServiceRegistry接口的實例).
3. 註冊事件;
4. 設置運行狀態
註冊:
org.springframework.cloud.client.serviceregistry.ServiceRegistry->註冊和取消註冊
->
EurekaServiceRegistry::register從參數獲取EurekaRegistration,
並初始化EurekaRegistration,經過ApplicationInfoManager設置實例的狀態,
若是有healthCheck則註冊healthCheck.
自動配置取消註冊:
1. 調用org.springframework.cloud.client.serviceregistry.ServiceRegistry接口的實例的deregister方法.
2. 設置狀態
取消註冊(EurekaServiceRegistry::deregister):
將狀態設置成DOWN.spa
ServiceInstance