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

com.alibaba.nacos.naming.core.Service Maven / Gradle / Ivy

There is a newer version: 3.4.0
Show newest version
/*
 * Copyright 1999-2018 Alibaba Group Holding Ltd.
 *
 * 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
 *
 *      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.alibaba.nacos.naming.core;

import com.alibaba.nacos.api.common.Constants;
import com.alibaba.nacos.api.selector.Selector;
import com.alibaba.nacos.common.utils.JacksonUtils;
import com.alibaba.nacos.common.utils.MD5Utils;
import com.alibaba.nacos.common.utils.StringUtils;
import com.alibaba.nacos.naming.consistency.KeyBuilder;
import com.alibaba.nacos.naming.consistency.RecordListener;
import com.alibaba.nacos.naming.core.v2.upgrade.doublewrite.delay.DoubleWriteEventListener;
import com.alibaba.nacos.naming.healthcheck.ClientBeatCheckTask;
import com.alibaba.nacos.naming.healthcheck.ClientBeatProcessor;
import com.alibaba.nacos.naming.healthcheck.HealthCheckReactor;
import com.alibaba.nacos.naming.healthcheck.RsInfo;
import com.alibaba.nacos.naming.misc.Loggers;
import com.alibaba.nacos.naming.misc.UtilsAndCommons;
import com.alibaba.nacos.naming.pojo.Record;
import com.alibaba.nacos.naming.push.UdpPushService;
import com.alibaba.nacos.naming.selector.NoneSelector;
import com.alibaba.nacos.sys.utils.ApplicationUtils;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.JsonInclude.Include;
import org.apache.commons.collections.CollectionUtils;
import org.apache.commons.collections.ListUtils;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

/**
 * Service of Nacos server side
 *
 * 

We introduce a 'service --> cluster --> instance' model, in which service stores a list of clusters, which * contain a list of instances. * *

his class inherits from Service in API module and stores some fields that do not have to expose to client. * * @author nkorange */ @JsonInclude(Include.NON_NULL) public class Service extends com.alibaba.nacos.api.naming.pojo.Service implements Record, RecordListener { private static final String SERVICE_NAME_SYNTAX = "[0-9a-zA-Z@\\.:_-]+"; @JsonIgnore private ClientBeatCheckTask clientBeatCheckTask = new ClientBeatCheckTask(this); /** * Identify the information used to determine how many isEmpty judgments the service has experienced. */ private int finalizeCount = 0; private String token; private List owners = new ArrayList<>(); private Boolean resetWeight = false; private Boolean enabled = true; private Selector selector = new NoneSelector(); private String namespaceId; /** * IP will be deleted if it has not send beat for some time, default timeout is 30 seconds. */ private long ipDeleteTimeout = 30 * 1000; private volatile long lastModifiedMillis = 0L; private volatile String checksum; /** * TODO set customized push expire time. */ private long pushCacheMillis = 0L; private Map clusterMap = new HashMap<>(); public Service() { } public Service(String name) { super(name); } @JsonIgnore public UdpPushService getPushService() { return ApplicationUtils.getBean(UdpPushService.class); } public long getIpDeleteTimeout() { return ipDeleteTimeout; } public void setIpDeleteTimeout(long ipDeleteTimeout) { this.ipDeleteTimeout = ipDeleteTimeout; } /** * Process client beat. * * @param rsInfo metrics info of server */ public void processClientBeat(final RsInfo rsInfo) { ClientBeatProcessor clientBeatProcessor = new ClientBeatProcessor(); clientBeatProcessor.setService(this); clientBeatProcessor.setRsInfo(rsInfo); HealthCheckReactor.scheduleNow(clientBeatProcessor); } public Boolean getEnabled() { return enabled; } public void setEnabled(Boolean enabled) { this.enabled = enabled; } public long getLastModifiedMillis() { return lastModifiedMillis; } public void setLastModifiedMillis(long lastModifiedMillis) { this.lastModifiedMillis = lastModifiedMillis; } public Boolean getResetWeight() { return resetWeight; } public void setResetWeight(Boolean resetWeight) { this.resetWeight = resetWeight; } public Selector getSelector() { return selector; } public void setSelector(Selector selector) { this.selector = selector; } @Override public boolean interests(String key) { return KeyBuilder.matchInstanceListKey(key, namespaceId, getName()); } @Override public boolean matchUnlistenKey(String key) { return KeyBuilder.matchInstanceListKey(key, namespaceId, getName()); } @Override public void onChange(String key, Instances value) throws Exception { Loggers.SRV_LOG.info("[NACOS-RAFT] datum is changed, key: {}, value: {}", key, value); for (Instance instance : value.getInstanceList()) { if (instance == null) { // Reject this abnormal instance list: throw new RuntimeException("got null instance " + key); } if (instance.getWeight() > 10000.0D) { instance.setWeight(10000.0D); } if (instance.getWeight() < 0.01D && instance.getWeight() > 0.0D) { instance.setWeight(0.01D); } } updateIPs(value.getInstanceList(), KeyBuilder.matchEphemeralInstanceListKey(key)); recalculateChecksum(); } @Override public void onDelete(String key) throws Exception { boolean isEphemeral = KeyBuilder.matchEphemeralInstanceListKey(key); for (Cluster each : clusterMap.values()) { each.updateIps(Collections.emptyList(), isEphemeral); } } /** * Get count of healthy instance in service. * * @return count of healthy instance */ public int healthyInstanceCount() { int healthyCount = 0; for (Instance instance : allIPs()) { if (instance.isHealthy()) { healthyCount++; } } return healthyCount; } public boolean triggerFlag() { return (healthyInstanceCount() * 1.0 / allIPs().size()) <= getProtectThreshold(); } /** * Update instances. * * @param instances instances * @param ephemeral whether is ephemeral instance */ public void updateIPs(Collection instances, boolean ephemeral) { Map> ipMap = new HashMap<>(clusterMap.size()); for (String clusterName : clusterMap.keySet()) { ipMap.put(clusterName, new ArrayList<>()); } for (Instance instance : instances) { try { if (instance == null) { Loggers.SRV_LOG.error("[NACOS-DOM] received malformed ip: null"); continue; } if (StringUtils.isEmpty(instance.getClusterName())) { instance.setClusterName(UtilsAndCommons.DEFAULT_CLUSTER_NAME); } if (!clusterMap.containsKey(instance.getClusterName())) { Loggers.SRV_LOG .warn("cluster: {} not found, ip: {}, will create new cluster with default configuration.", instance.getClusterName(), instance.toJson()); Cluster cluster = new Cluster(instance.getClusterName(), this); cluster.init(); getClusterMap().put(instance.getClusterName(), cluster); } List clusterIPs = ipMap.get(instance.getClusterName()); if (clusterIPs == null) { clusterIPs = new LinkedList<>(); ipMap.put(instance.getClusterName(), clusterIPs); } clusterIPs.add(instance); } catch (Exception e) { Loggers.SRV_LOG.error("[NACOS-DOM] failed to process ip: " + instance, e); } } for (Map.Entry> entry : ipMap.entrySet()) { //make every ip mine List entryIPs = entry.getValue(); clusterMap.get(entry.getKey()).updateIps(entryIPs, ephemeral); } setLastModifiedMillis(System.currentTimeMillis()); getPushService().serviceChanged(this); ApplicationUtils.getBean(DoubleWriteEventListener.class).doubleWriteToV2(this, ephemeral); StringBuilder stringBuilder = new StringBuilder(); for (Instance instance : allIPs()) { stringBuilder.append(instance.toIpAddr()).append('_').append(instance.isHealthy()).append(','); } Loggers.EVT_LOG.info("[IP-UPDATED] namespace: {}, service: {}, ips: {}", getNamespaceId(), getName(), stringBuilder.toString()); } /** * Init service. */ public void init() { HealthCheckReactor.scheduleCheck(clientBeatCheckTask); for (Map.Entry entry : clusterMap.entrySet()) { entry.getValue().setService(this); entry.getValue().init(); } } /** * Destroy service. * * @throws Exception exception */ public void destroy() throws Exception { for (Map.Entry entry : clusterMap.entrySet()) { entry.getValue().destroy(); } HealthCheckReactor.cancelCheck(clientBeatCheckTask); ApplicationUtils.getBean(DoubleWriteEventListener.class) .doubleWriteMetadataToV2(this, this.allIPs(false).isEmpty(), true); } /** * Judge whether service has instance. * * @return true if no instance, otherwise false */ public boolean isEmpty() { for (Map.Entry entry : clusterMap.entrySet()) { final Cluster cluster = entry.getValue(); if (!cluster.isEmpty()) { return false; } } return true; } /** * Get all instance. * * @return list of all instance */ public List allIPs() { List result = new ArrayList<>(); for (Map.Entry entry : clusterMap.entrySet()) { result.addAll(entry.getValue().allIPs()); } return result; } /** * Get all instance of ephemeral or consistency. * * @param ephemeral whether ephemeral instance * @return all instance of ephemeral if @param ephemeral = true, otherwise all instance of consistency */ public List allIPs(boolean ephemeral) { List result = new ArrayList<>(); for (Map.Entry entry : clusterMap.entrySet()) { result.addAll(entry.getValue().allIPs(ephemeral)); } return result; } /** * Get all instance from input clusters. * * @param clusters cluster names * @return all instance from input clusters. */ public List allIPs(List clusters) { List result = new ArrayList<>(); for (String cluster : clusters) { Cluster clusterObj = clusterMap.get(cluster); if (clusterObj == null) { continue; } result.addAll(clusterObj.allIPs()); } return result; } /** * Get all instance from input clusters. * * @param clusters cluster names * @return all instance from input clusters, if clusters is empty, return all cluster */ public List srvIPs(List clusters) { if (CollectionUtils.isEmpty(clusters)) { clusters = new ArrayList<>(); clusters.addAll(clusterMap.keySet()); } return allIPs(clusters); } public String toJson() { return JacksonUtils.toJson(this); } @JsonIgnore public String getServiceString() { Map serviceObject = new HashMap(10); Service service = this; serviceObject.put("name", service.getName()); List ips = service.allIPs(); int invalidIpCount = 0; int ipCount = 0; for (Instance ip : ips) { if (!ip.isHealthy()) { invalidIpCount++; } ipCount++; } serviceObject.put("ipCount", ipCount); serviceObject.put("invalidIPCount", invalidIpCount); serviceObject.put("owners", service.getOwners()); serviceObject.put("token", service.getToken()); serviceObject.put("protectThreshold", service.getProtectThreshold()); List clustersList = new ArrayList(); for (Map.Entry entry : service.getClusterMap().entrySet()) { Cluster cluster = entry.getValue(); Map clusters = new HashMap(10); clusters.put("name", cluster.getName()); clusters.put("healthChecker", cluster.getHealthChecker()); clusters.put("defCkport", cluster.getDefCkport()); clusters.put("defIPPort", cluster.getDefIPPort()); clusters.put("useIPPort4Check", cluster.isUseIPPort4Check()); clusters.put("sitegroup", cluster.getSitegroup()); clustersList.add(clusters); } serviceObject.put("clusters", clustersList); try { return JacksonUtils.toJson(serviceObject); } catch (Exception e) { throw new RuntimeException("Service toJson failed", e); } } public String getToken() { return token; } public void setToken(String token) { this.token = token; } public List getOwners() { return owners; } public void setOwners(List owners) { this.owners = owners; } public Map getClusterMap() { return clusterMap; } public void setClusterMap(Map clusterMap) { this.clusterMap = clusterMap; } public String getNamespaceId() { return namespaceId; } public void setNamespaceId(String namespaceId) { this.namespaceId = namespaceId; } /** * Update from other service. * * @param vDom other service */ public void update(Service vDom) { if (!StringUtils.equals(token, vDom.getToken())) { Loggers.SRV_LOG.info("[SERVICE-UPDATE] service: {}, token: {} -> {}", getName(), token, vDom.getToken()); token = vDom.getToken(); } if (!ListUtils.isEqualList(owners, vDom.getOwners())) { Loggers.SRV_LOG.info("[SERVICE-UPDATE] service: {}, owners: {} -> {}", getName(), owners, vDom.getOwners()); owners = vDom.getOwners(); } if (getProtectThreshold() != vDom.getProtectThreshold()) { Loggers.SRV_LOG .info("[SERVICE-UPDATE] service: {}, protectThreshold: {} -> {}", getName(), getProtectThreshold(), vDom.getProtectThreshold()); setProtectThreshold(vDom.getProtectThreshold()); } if (resetWeight != vDom.getResetWeight().booleanValue()) { Loggers.SRV_LOG.info("[SERVICE-UPDATE] service: {}, resetWeight: {} -> {}", getName(), resetWeight, vDom.getResetWeight()); resetWeight = vDom.getResetWeight(); } if (enabled != vDom.getEnabled().booleanValue()) { Loggers.SRV_LOG .info("[SERVICE-UPDATE] service: {}, enabled: {} -> {}", getName(), enabled, vDom.getEnabled()); enabled = vDom.getEnabled(); } selector = vDom.getSelector(); setMetadata(vDom.getMetadata()); updateOrAddCluster(vDom.getClusterMap().values()); remvDeadClusters(this, vDom); Loggers.SRV_LOG.info("cluster size, new: {}, old: {}", getClusterMap().size(), vDom.getClusterMap().size()); recalculateChecksum(); ApplicationUtils.getBean(DoubleWriteEventListener.class) .doubleWriteMetadataToV2(this, vDom.allIPs(false).isEmpty(), false); } @Override public String getChecksum() { if (StringUtils.isEmpty(checksum)) { recalculateChecksum(); } return checksum; } /** * Re-calculate checksum of service. */ public synchronized void recalculateChecksum() { List ips = allIPs(); StringBuilder ipsString = new StringBuilder(); ipsString.append(getServiceString()); if (Loggers.SRV_LOG.isDebugEnabled()) { Loggers.SRV_LOG.debug("service to json: " + getServiceString()); } if (CollectionUtils.isNotEmpty(ips)) { Collections.sort(ips); } for (Instance ip : ips) { String string = ip.getIp() + ":" + ip.getPort() + "_" + ip.getWeight() + "_" + ip.isHealthy() + "_" + ip .getClusterName(); ipsString.append(string); ipsString.append(','); } checksum = MD5Utils.md5Hex(ipsString.toString(), Constants.ENCODE); } private void updateOrAddCluster(Collection clusters) { for (Cluster cluster : clusters) { Cluster oldCluster = clusterMap.get(cluster.getName()); if (oldCluster != null) { oldCluster.setService(this); oldCluster.update(cluster); } else { cluster.init(); cluster.setService(this); clusterMap.put(cluster.getName(), cluster); } } } private void remvDeadClusters(Service oldDom, Service newDom) { Collection oldClusters = oldDom.getClusterMap().values(); Collection newClusters = newDom.getClusterMap().values(); List deadClusters = (List) CollectionUtils.subtract(oldClusters, newClusters); for (Cluster cluster : deadClusters) { oldDom.getClusterMap().remove(cluster.getName()); cluster.destroy(); } } public int getFinalizeCount() { return finalizeCount; } public void setFinalizeCount(int finalizeCount) { this.finalizeCount = finalizeCount; } public void addCluster(Cluster cluster) { clusterMap.put(cluster.getName(), cluster); } /** * Judge whether service is validate. * * @throws IllegalArgumentException if service is not validate */ public void validate() { if (!getName().matches(SERVICE_NAME_SYNTAX)) { throw new IllegalArgumentException( "dom name can only have these characters: 0-9a-zA-Z-._:, current: " + getName()); } for (Cluster cluster : clusterMap.values()) { cluster.validate(); } } }