国产无遮挡裸体免费直播视频,久久精品国产蜜臀av,动漫在线视频一区二区,欧亚日韩一区二区三区,久艹在线 免费视频,国产精品美女网站免费,正在播放 97超级视频在线观看,斗破苍穹年番在线观看免费,51最新乱码中文字幕

nacos注冊(cè)中心單節(jié)點(diǎn)ap架構(gòu)源碼解析(最新推薦)

 更新時(shí)間:2023年01月03日 09:31:42   作者:bei_er  
這篇文章主要介紹了nacos注冊(cè)中心單節(jié)點(diǎn)ap架構(gòu)源碼解析,本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下

一、注冊(cè)流程

單nacos節(jié)點(diǎn)流程圖如下:

流程圖可以知,Nacos注冊(cè)流程包括客戶端的服務(wù)注冊(cè)、服務(wù)實(shí)例列表拉取、定時(shí)心跳任務(wù);以及服務(wù)端的定時(shí)檢查服務(wù)實(shí)例任務(wù)、服務(wù)實(shí)例更新推送5個(gè)功能。

服務(wù)注冊(cè):當(dāng)客戶端啟動(dòng)的時(shí)候會(huì)根據(jù)當(dāng)前微服務(wù)的配置信息把微服務(wù)注冊(cè)到nacos服務(wù)端。

服務(wù)實(shí)例列表拉取:當(dāng)客戶端啟動(dòng)的時(shí)候從nacos服務(wù)端獲取當(dāng)前服務(wù)的名稱已經(jīng)注冊(cè)的實(shí)例數(shù)據(jù),并把這些實(shí)例數(shù)據(jù)緩存在客戶端的serviceInfoMap 對(duì)象中。

定時(shí)心跳任務(wù):當(dāng)客戶端向nacos服務(wù)注冊(cè)臨時(shí)實(shí)例對(duì)象的時(shí)候,會(huì)創(chuàng)建一個(gè)延期的任務(wù)去往服務(wù)端發(fā)送心跳信息。如果發(fā)送心跳信息成功,則又會(huì)創(chuàng)建一個(gè)延期任務(wù)往服務(wù)端注冊(cè)心跳信息,一直重復(fù)該邏輯。nacos服務(wù)端接收到客戶端的心跳信息就是更新客戶端實(shí)例的最后心跳時(shí)間。該時(shí)間用來判斷實(shí)例是否健康和是否需要?jiǎng)h除。

定時(shí)檢查服務(wù)實(shí)例任務(wù):nacos服務(wù)端在創(chuàng)建空服務(wù)對(duì)象的時(shí)候會(huì)通過線程池來定時(shí)執(zhí)行檢查服務(wù),其主要邏輯為判斷當(dāng)前時(shí)間和最后心跳時(shí)間之差是否大于健康超時(shí)時(shí)間和刪除實(shí)例超時(shí)時(shí)間,如果大于,則更新實(shí)例的健康狀態(tài)和刪除當(dāng)前實(shí)例。定時(shí)執(zhí)行的規(guī)則為5秒之后執(zhí)行檢查,并且每次執(zhí)行完檢查之后,5秒之后再次執(zhí)行檢查。

服務(wù)實(shí)例更新推送:當(dāng)有客戶端更新實(shí)例對(duì)象時(shí),服務(wù)端會(huì)先獲取該客戶端的服務(wù)名稱下所有已經(jīng)注冊(cè)的客戶端實(shí)例,并會(huì)針每一個(gè)客戶端發(fā)送一個(gè)更新serviceinfo的udp消息,客戶端監(jiān)聽收到nacos服務(wù)端發(fā)送的udp數(shù)據(jù)后進(jìn)行本地緩存的更新。

二、客戶端

一、服務(wù)注冊(cè)

根據(jù)spring-cloud-starter-alibaba-nacos-discovery的spring.factories文件,找到服務(wù)注冊(cè)啟動(dòng)配置類。

spring.factories文件內(nèi)容為如下,

org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
  com.alibaba.cloud.nacos.discovery.NacosDiscoveryAutoConfiguration,\
  com.alibaba.cloud.nacos.ribbon.RibbonNacosAutoConfiguration,\
  com.alibaba.cloud.nacos.endpoint.NacosDiscoveryEndpointAutoConfiguration,\
  com.alibaba.cloud.nacos.registry.NacosServiceRegistryAutoConfiguration,\
  com.alibaba.cloud.nacos.discovery.NacosDiscoveryClientConfiguration,\
  com.alibaba.cloud.nacos.discovery.reactive.NacosReactiveDiscoveryClientConfiguration,\
  com.alibaba.cloud.nacos.discovery.configclient.NacosConfigServerAutoConfiguration,\
  com.alibaba.cloud.nacos.NacosServiceAutoConfiguration
org.springframework.cloud.bootstrap.BootstrapConfiguration=\
  com.alibaba.cloud.nacos.discovery.configclient.NacosDiscoveryClientConfigServiceBootstrapConfiguration

根據(jù)名稱判斷可以得出 NacosServiceRegistryAutoConfiguration 為服務(wù)注冊(cè)啟動(dòng)配置類,源碼如下

@Configuration(proxyBeanMethods = false)
@EnableConfigurationProperties
@ConditionalOnNacosDiscoveryEnabled
@ConditionalOnProperty(value = "spring.cloud.service-registry.auto-registration.enabled",
		matchIfMissing = true)
@AutoConfigureAfter({ AutoServiceRegistrationConfiguration.class,
		AutoServiceRegistrationAutoConfiguration.class,
		NacosDiscoveryAutoConfiguration.class })
public class NacosServiceRegistryAutoConfiguration {

	@Bean
	public NacosServiceRegistry nacosServiceRegistry(
			NacosDiscoveryProperties nacosDiscoveryProperties) {
		return new NacosServiceRegistry(nacosDiscoveryProperties);
	}

	@Bean
	@ConditionalOnBean(AutoServiceRegistrationProperties.class)
	public NacosRegistration nacosRegistration(
			ObjectProvider<List<NacosRegistrationCustomizer>> registrationCustomizers,
			NacosDiscoveryProperties nacosDiscoveryProperties,
			ApplicationContext context) {
		return new NacosRegistration(registrationCustomizers.getIfAvailable(),
				nacosDiscoveryProperties, context);
	}

	@Bean
	@ConditionalOnBean(AutoServiceRegistrationProperties.class)
	public NacosAutoServiceRegistration nacosAutoServiceRegistration(
			NacosServiceRegistry registry,
			AutoServiceRegistrationProperties autoServiceRegistrationProperties,
			NacosRegistration registration) {
		return new NacosAutoServiceRegistration(registry,
				autoServiceRegistrationProperties, registration);
	}

關(guān)鍵類 NacosAutoServiceRegistration 的類圖結(jié)構(gòu)如下

上圖可知,NacosAutoServiceRegistration 實(shí)現(xiàn)了 ApplicationListener接口,該監(jiān)聽器會(huì)在SpringBoot啟動(dòng)的時(shí)候會(huì)自動(dòng)調(diào)用 onApplicationEvent方法,onApplicationEvent具體實(shí)現(xiàn)方法如下

public void onApplicationEvent(WebServerInitializedEvent event) {
    this.bind(event);
}

@Deprecated
public void bind(WebServerInitializedEvent event) {
    ApplicationContext context = event.getApplicationContext();
    if (!(context instanceof ConfigurableWebServerApplicationContext) || !"management".equals(((ConfigurableWebServerApplicationContext)context).getServerNamespace())) {
        this.port.compareAndSet(0, event.getWebServer().getPort());
        // 具體的啟動(dòng)方法
        this.start();
    }
}

具體的啟動(dòng)方法this.start();方法的代碼如下,

public void start() {
    if (!this.isEnabled()) {
        if (logger.isDebugEnabled()) {
            logger.debug("Discovery Lifecycle disabled. Not starting");
        }

    } else {
        if (!this.running.get()) {
            this.context.publishEvent(new InstancePreRegisteredEvent(this, this.getRegistration()));
            // 關(guān)鍵邏輯
            this.register();
            if (this.shouldRegisterManagement()) {
                this.registerManagement();
            }

            this.context.publishEvent(new InstanceRegisteredEvent(this, this.getConfiguration()));
            this.running.compareAndSet(false, true);
        }

    }

關(guān)鍵邏輯為this.register();方法代碼如下

protected void register() {
    if (!this.registration.getNacosDiscoveryProperties().isRegisterEnabled()) {
        log.debug("Registration disabled.");
        return;
    }
    if (this.registration.getPort() < 0) {
        this.registration.setPort(getPort().get());
    }
    super.register();
}

關(guān)鍵邏輯為super.register();方法代碼如下,

protected void register() {
    this.serviceRegistry.register(this.getRegistration());
}

關(guān)鍵邏輯為this.serviceRegistry.register方法代碼如下,

@Override
public void register(Registration registration) {

    if (StringUtils.isEmpty(registration.getServiceId())) {
        log.warn("No service to register for nacos client...");
        return;
    }
	// 根據(jù)配置屬性構(gòu)建NamingService對(duì)象
    NamingService namingService = namingService();
    // 獲取服務(wù)名,默認(rèn)為 ${spring.application.name}
    String serviceId = registration.getServiceId();
    // 獲取組名 ,默認(rèn)為 DEFAULT_GROUP
    String group = nacosDiscoveryProperties.getGroup();

    // 創(chuàng)建注冊(cè)實(shí)例
    Instance instance = getNacosInstanceFromRegistration(registration);

    try {
        // 發(fā)起注冊(cè)
        namingService.registerInstance(serviceId, group, instance);
        log.info("nacos registry, {} {} {}:{} register finished", group, serviceId,
                 instance.getIp(), instance.getPort());
    }
    catch (Exception e) {
        log.error("nacos registry, {} register failed...{},", serviceId,
                  registration.toString(), e);
        // rethrow a RuntimeException if the registration is failed.
        // issue : https://github.com/alibaba/spring-cloud-alibaba/issues/1132
        rethrowRuntimeException(e);
    }
}

先通過getNacosInstanceFromRegistration方法創(chuàng)建實(shí)例對(duì)象,getNacosInstanceFromRegistration代碼如下,

private Instance getNacosInstanceFromRegistration(Registration registration) {
    Instance instance = new Instance();
    // 獲取服務(wù)ip
    instance.setIp(registration.getHost());
    // 獲取服務(wù)
    instance.setPort(registration.getPort());
    // 獲取權(quán)重
    instance.setWeight(nacosDiscoveryProperties.getWeight());
    // 獲取集群名稱
    instance.setClusterName(nacosDiscoveryProperties.getClusterName());
  
    instance.setEnabled(nacosDiscoveryProperties.isInstanceEnabled());
    // 獲取元數(shù)據(jù)
    instance.setMetadata(registration.getMetadata());
    // 獲取是否為臨時(shí)實(shí)例
    instance.setEphemeral(nacosDiscoveryProperties.isEphemeral());
    return instance;
}

然后通過namingService.registerInstance方法發(fā)起注冊(cè),registerInstance方法的代碼如下,

public void registerInstance(String serviceName, String groupName, Instance instance) throws NacosException {
    // 檢查 實(shí)例是否合法 
    // heart beat timeout must(默認(rèn)15秒) < heart beat interval (默認(rèn)5秒)拋異常
    // ip delete timeout must(默認(rèn)30 秒) < heart beat interval(默認(rèn)5秒)拋異常
    NamingUtils.checkInstanceIsLegal(instance);
    // 構(gòu)建 groupName@@serviceName
    String groupedServiceName = NamingUtils.getGroupedName(serviceName, groupName);
    // 如果是臨時(shí)實(shí)例,則創(chuàng)建心跳信息,定時(shí)給nacos服務(wù)發(fā)送
    if (instance.isEphemeral()) {
        BeatInfo beatInfo = this.beatReactor.buildBeatInfo(groupedServiceName, instance);
        this.beatReactor.addBeatInfo(groupedServiceName, beatInfo);
    }
	// 向 nacos-service 注冊(cè)實(shí)例
    this.serverProxy.registerService(groupedServiceName, groupName, instance);
}

先檢查實(shí)例是否合法,然后構(gòu)建服務(wù)名稱,規(guī)則為groupName@@serviceName。通過this.serverProxy.registerService方法向 nacos-service 注冊(cè)實(shí)例,代碼如下,

public void registerService(String serviceName, String groupName, Instance instance) throws NacosException {

    NAMING_LOGGER.info("[REGISTER-SERVICE] {} registering service {} with instance: {}", namespaceId, serviceName,instance);

    final Map<String, String> params = new HashMap<String, String>(16);
    //設(shè)置 namespaceId
    params.put(CommonParams.NAMESPACE_ID, namespaceId);
    //設(shè)置 serviceName
    params.put(CommonParams.SERVICE_NAME, serviceName);
    //設(shè)置 groupName
    params.put(CommonParams.GROUP_NAME, groupName);
    //設(shè)置 clusterName
    params.put(CommonParams.CLUSTER_NAME, instance.getClusterName());
    params.put("ip", instance.getIp());
    params.put("port", String.valueOf(instance.getPort()));
    params.put("weight", String.valueOf(instance.getWeight()));
    params.put("enable", String.valueOf(instance.isEnabled()));
    params.put("healthy", String.valueOf(instance.isHealthy()));
    params.put("ephemeral", String.valueOf(instance.isEphemeral()));
    params.put("metadata", JacksonUtils.toJson(instance.getMetadata()));
	// 調(diào)用 nacos-service 的nacosUrlInstance接口注冊(cè)實(shí)例
    reqApi(UtilAndComs.nacosUrlInstance, params, HttpMethod.POST);

}

通過向reqApi方法向nacos服務(wù)端注冊(cè)當(dāng)前實(shí)例數(shù)據(jù),其實(shí)就是向 ${spring.cloud.nacos.discovery.server-addr}/nacos/v1/ns/instance 發(fā)送POST請(qǐng)求。該請(qǐng)求地址對(duì)應(yīng)的nacos服務(wù)端的源碼的naming工程中InstanceController的register方法,代碼如下,

public String register(HttpServletRequest request) throws Exception {
    final String namespaceId = WebUtils
        .optional(request, CommonParams.NAMESPACE_ID, Constants.DEFAULT_NAMESPACE_ID);
    final String serviceName = WebUtils.required(request, CommonParams.SERVICE_NAME);
    NamingUtils.checkServiceNameFormat(serviceName);
	//根據(jù)請(qǐng)求構(gòu)建 Instance 對(duì)象
    final Instance instance = parseInstance(request);
	//注冊(cè) Instance 對(duì)象,serviceManager對(duì)象中保存了所有的服務(wù)對(duì)象。
    serviceManager.registerInstance(namespaceId, serviceName, instance);
    return "ok";
}

先根據(jù)請(qǐng)求對(duì)象構(gòu)建Instance對(duì)象,然后通過serviceManager.registerInstance方法用來注冊(cè)Instance對(duì)象,registerInstance代碼如下

public void registerInstance(String namespaceId, String serviceName, Instance instance) throws NacosException {
	// 如果 namespaceId 為 key 的數(shù)據(jù)為空,則創(chuàng)建 service ,并初始化service
    createEmptyService(namespaceId, serviceName, instance.isEphemeral());
	// 獲取 service 對(duì)象
    Service service = getService(namespaceId, serviceName);
	// 如果 service為空 則報(bào)錯(cuò)
    if (service == null) {
        throw new NacosException(NacosException.INVALID_PARAM,
                                 "service not found, namespace: " + namespaceId + ", service: " + serviceName);
    }
	// 添加實(shí)例
    addInstance(namespaceId, serviceName, instance.isEphemeral(), instance);
}

如果 namespaceId為key的數(shù)據(jù)為空,則創(chuàng)建 service,并初始化service。然后調(diào)用addInstance添加實(shí)例對(duì)象,addInstance方法代碼如下,

public void addInstance(String namespaceId, String serviceName, boolean ephemeral, Instance... ips)
    throws NacosException {
	  // 根據(jù) 命名空間 和 服務(wù)名稱 構(gòu)建 key
        String key = KeyBuilder.buildInstanceListKey(namespaceId, serviceName, ephemeral);
        // 獲取 service
        Service service = getService(namespaceId, serviceName);
        // 同步鎖
        synchronized (service) {
            // 獲取服務(wù)下的實(shí)例集合(服務(wù)已有 + 新增的實(shí)例)
            List<Instance> instanceList = addIpAddresses(service, ephemeral, ips);
            Instances instances = new Instances();
            instances.setInstanceList(instanceList);
            // 根據(jù)KEY添加服務(wù)的實(shí)例
            consistencyService.put(key, instances);
        }
}

addIpAddresses方法中會(huì)調(diào)用updateIpAddresses方法,且action為 add。該方法根據(jù)action的值來獲取該服務(wù)下的最新實(shí)例集合(新增實(shí)例或刪除實(shí)例加上目前服務(wù)已有的實(shí)例數(shù)據(jù)合集)。如果action為add表示新增,則方法最后返回的集合對(duì)象中會(huì)把該服務(wù)中已有的實(shí)例集合加上新增的實(shí)例集合數(shù)據(jù)一起返回 ;如果action為 remove表示刪除,則方法最后返回的集合對(duì)象中會(huì)把該服務(wù)中已有的實(shí)例集合刪除掉需要?jiǎng)h除的實(shí)例集合數(shù)據(jù)。后面通過調(diào)用consistencyService.put(key, instances)方法來把updateIpAddresses方法返回的值直接添加consistencyService的實(shí)例中。updateIpAddresses方法的代碼如下,

public List<Instance> updateIpAddresses(Service service, String action, boolean ephemeral, Instance... ips)
    throws NacosException {
    // 從本地緩存中獲取服務(wù)的實(shí)例數(shù)據(jù)
    Datum datum = consistencyService
        .get(KeyBuilder.buildInstanceListKey(service.getNamespaceId(), service.getName(), ephemeral));
    // 獲取 當(dāng)前服務(wù)下所有的 實(shí)例
    List<Instance> currentIPs = service.allIPs(ephemeral);
    // 創(chuàng)建當(dāng)前實(shí)例數(shù)據(jù)map
    Map<String, Instance> currentInstances = new HashMap<>(currentIPs.size());
    // 創(chuàng)建 當(dāng)前實(shí)例Id set
    Set<String> currentInstanceIds = Sets.newHashSet();

    // 遍歷當(dāng)前服務(wù)的所有實(shí)例,添加到 創(chuàng)建當(dāng)前實(shí)例數(shù)據(jù) map 和 當(dāng)前實(shí)例Id集合
    for (Instance instance : currentIPs) {
        currentInstances.put(instance.toIpAddr(), instance);
        currentInstanceIds.add(instance.getInstanceId());
    }
    // 構(gòu)造 實(shí)例集合對(duì)象的 map
    Map<String, Instance> instanceMap;
    // 如果有緩存數(shù)據(jù)
    if (datum != null && null != datum.value) {
        // 從本地緩存中以及當(dāng)前服務(wù)的內(nèi)存數(shù)據(jù)獲取最新服務(wù)的實(shí)例數(shù)據(jù)
        instanceMap = setValid(((Instances) datum.value).getInstanceList(), currentInstances);
    }
    // 如果沒有緩存數(shù)據(jù)
    else {
        // 創(chuàng)建 instanceMap
        instanceMap = new HashMap<>(ips.length);
    }
    // 遍歷參數(shù)傳過來的實(shí)例對(duì)象
    for (Instance instance : ips) {
        // 如果 service 不包括 實(shí)例的 ClusterName 則創(chuàng)建 實(shí)例 Cluster,并初始化
        if (!service.getClusterMap().containsKey(instance.getClusterName())) {
            Cluster cluster = new Cluster(instance.getClusterName(), service);
            cluster.init();
            service.getClusterMap().put(instance.getClusterName(), cluster);
            Loggers.SRV_LOG
                .warn("cluster: {} not found, ip: {}, will create new cluster with default configuration.",
                      instance.getClusterName(), instance.toJson());
        }
        // 如果是刪除,則從 instanceMap 中 刪除 該實(shí)例
        if (UtilsAndCommons.UPDATE_INSTANCE_ACTION_REMOVE.equals(action)) {
            instanceMap.remove(instance.getDatumKey());
        }
        // 如果是新增
        else {
            //獲取已存在的 實(shí)例
            Instance oldInstance = instanceMap.get(instance.getDatumKey());
            if (oldInstance != null) {
                instance.setInstanceId(oldInstance.getInstanceId());
            } else {
                // 生成 實(shí)例 id
                instance.setInstanceId(instance.generateInstanceId(currentInstanceIds));
            }
            // instanceMap 添加instance實(shí)例
            instanceMap.put(instance.getDatumKey(), instance);
        }

    }
    // 如果集合小于0 ,并且是新增操作則拋異常
    if (instanceMap.size() <= 0 && UtilsAndCommons.UPDATE_INSTANCE_ACTION_ADD.equals(action)) {
        throw new IllegalArgumentException(
            "ip list can not be empty, service: " + service.getName() + ", ip list: " + JacksonUtils
            .toJson(instanceMap.values()));
    }
    // 返回 服務(wù)中最新的實(shí)例數(shù)據(jù)
    return new CopyOnWriteArrayList<>(instanceMap.values());
}

通過updateIpAddresses方法拿到需要更新的實(shí)例集合對(duì)象后,再通過consistencyService.put(key, instances)把拿到的實(shí)例集合對(duì)象添加到實(shí)現(xiàn)了PersistentConsistencyServiceDelegateImpl或者EphemeralConsistencyService接口的實(shí)例對(duì)象中,consistencyService.put(key, instances)的源碼如下,

@Override
public void put(String key, Record value) throws NacosException {
    // 根據(jù)key獲取具體的 consistencyService ,并且向其中添加具體的 key 和 value
    mapConsistencyService(key).put(key, value);
}

根據(jù)key獲取具體的 consistencyService ,并且向其中添加具體的 key 和 value。consistencyService中根據(jù)key獲取集群的實(shí)例對(duì)象(臨時(shí)服務(wù)對(duì)象EphemeralConsistencyService和持久服務(wù)對(duì)象PersistentConsistencyServiceDelegateImpl)

private ConsistencyService mapConsistencyService(String key) {
    // 根據(jù)key返回具體的服務(wù)對(duì)象
    return KeyBuilder.matchEphemeralKey(key) ? ephemeralConsistencyService : persistentConsistencyService;
}

如果是注冊(cè)的臨時(shí)實(shí)例節(jié)點(diǎn),這里取到的是實(shí)現(xiàn)了ephemeralConsistencyService接口的DistroConsistencyServiceImpl 對(duì)象,它的put源碼如下:

@Override
public void put(String key, Record value) throws NacosException {
    // 添加key 和 value
    onPut(key, value);
    distroProtocol.sync(new DistroKey(key, KeyBuilder.INSTANCE_LIST_KEY_PREFIX), DataOperation.CHANGE,
                        globalConfig.getTaskDispatchPeriod() / 2);
}

通過onPut方法添加key 和 value,opPut方法的代碼如下,

public void onPut(String key, Record value) {
    // 如果是臨時(shí)節(jié)點(diǎn)實(shí)例,則創(chuàng)建 Datum 并保存在 dataStore 中
    if (KeyBuilder.matchEphemeralInstanceListKey(key)) {
        Datum<Instances> datum = new Datum<>();
        datum.value = (Instances) value;
        datum.key = key;
        datum.timestamp.incrementAndGet();
        dataStore.put(key, datum);
    }
    // 如果 監(jiān)聽對(duì)象不包括 key 則返回
    if (!listeners.containsKey(key)) {
        return;
    }
    // 向notifier對(duì)象添加通知任務(wù)
    notifier.addTask(key, DataOperation.CHANGE);
}

如果是臨時(shí)實(shí)例節(jié)點(diǎn),則創(chuàng)建 Datum 并保存在 dataStore 中,然后通過notifier.addTask用來向notifier對(duì)象添加通知任務(wù),且操作類型為DataOperation.CHANGE,addTask方法的代碼如下:

public void addTask(String datumKey, DataOperation action) {
    // 如果services包括了當(dāng)前的 datumKey ,并且是修改操作 則直接返回
    if (services.containsKey(datumKey) && action == DataOperation.CHANGE) {
        return;
    }
    // 如果是修改操作,則向 services 添加 datumKey
    if (action == DataOperation.CHANGE) {
        services.put(datumKey, StringUtils.EMPTY);
    }
    // 向 tasks中添加 Pair 對(duì)象
    tasks.offer(Pair.with(datumKey, action));
}

以上代碼的tasks是用來存放具體實(shí)例key和動(dòng)作類型的對(duì)象,它是一個(gè)ArrayBlockingQueue對(duì)象,DistroConsistencyServiceImpl 對(duì)象的init方法代碼如下,

@PostConstruct
public void init() {
    GlobalExecutor.submitDistroNotifyTask(notifier);
}

根據(jù)以上代碼可知,在DistroConsistencyServiceImpl 實(shí)例對(duì)象初始化之后會(huì)往GlobalExecutor線程池對(duì)象中添加了一個(gè)notifier對(duì)象。notifier對(duì)象為一個(gè)實(shí)現(xiàn)了Runnable 的實(shí)例。上面的代碼會(huì)執(zhí)行notifier對(duì)象的run方法,notifier的run方法代碼如下:

public void run() {
    Loggers.DISTRO.info("distro notifier started");
    // 死循環(huán)遍歷
    for (; ; ) {
        try {
            // 獲取 tasks的數(shù)據(jù),如果沒有數(shù)據(jù)會(huì)阻塞當(dāng)前線程,直到tasks有數(shù)據(jù)為止。
            Pair<String, DataOperation> pair = tasks.take();
            // 處理數(shù)據(jù)
            handle(pair);
        } catch (Throwable e) {
            Loggers.DISTRO.error("[NACOS-DISTRO] Error while handling notifying task", e);
        }
    }
}

上面是一個(gè)死循環(huán),tasks.take()是一個(gè)阻塞式獲取數(shù)據(jù)的方法,如果tasks沒有數(shù)據(jù)則會(huì)阻塞當(dāng)前線程直到tasks.take()拿到數(shù)據(jù),拿到數(shù)據(jù)之后會(huì)調(diào)用handle方法處理,handle代碼如下,

private void handle(Pair<String, DataOperation> pair) {
    try {
        String datumKey = pair.getValue0();
        DataOperation action = pair.getValue1();
        // 先從 services 中刪除 key
        services.remove(datumKey);

        int count = 0;
        // 根據(jù) key 獲取 服務(wù)對(duì)象數(shù)據(jù)
        ConcurrentLinkedQueue<RecordListener> recordListeners = listeners.get(datumKey);
        if (recordListeners == null) {
            Loggers.DISTRO.info("[DISTRO-WARN] RecordListener not found, key: {}", datumKey);
            return;
        }
        for (RecordListener listener : recordListeners) {
            count++;
            try {
                // 如果是新增
                if (action == DataOperation.CHANGE) {
                    Datum datum = dataStore.get(datumKey);
                    if (datum != null) {
                        // 更新 serivce 的實(shí)例數(shù)據(jù)
                        listener.onChange(datumKey, datum.value);
                    } else {
                        Loggers.DISTRO.info("[DISTRO-WARN] data not found, key: {}", datumKey);
                    }
                    continue;
                }
                // 如果是刪除
                if (action == DataOperation.DELETE) {
                    listener.onDelete(datumKey);
                    continue;
                }
            } catch (Throwable e) {
                Loggers.DISTRO.error("[NACOS-DISTRO] error while notifying listener of key: {}", datumKey, e);
            }
        }

        if (Loggers.DISTRO.isDebugEnabled()) {
            Loggers.DISTRO
                .debug("[NACOS-DISTRO] datum change notified, key: {}, listener count: {}, action: {}",
                       datumKey, count, action.name());
        }
    } catch (Throwable e) {
        Loggers.DISTRO.error("[NACOS-DISTRO] Error while handling notifying task", e);
    }
}

根據(jù)action 為 DataOperation.CHANGE,代碼中執(zhí)行的代碼分支為listener.onChange(datumKey, datum.value),該方法的邏輯為修改服務(wù)的實(shí)例數(shù)據(jù),源碼如下

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);
        }
    }
    // 更新 service 的 實(shí)例集合
    updateIPs(value.getInstanceList(), KeyBuilder.matchEphemeralInstanceListKey(key));

    recalculateChecksum();
}

以上代碼先遍歷所有的實(shí)例數(shù)據(jù)設(shè)置權(quán)值,再通過updateIPs方法更新服務(wù)實(shí)例,updateIPs方法的代碼如下:

public void updateIPs(Collection<Instance> instances, boolean ephemeral) {
    // 根據(jù) clusterMap 創(chuàng)建 ipMap對(duì)象
    Map<String, List<Instance>> ipMap = new HashMap<>(clusterMap.size());
    // 根據(jù) clusterMap 初始化 ipMap對(duì)象
    for (String clusterName : clusterMap.keySet()) {
        ipMap.put(clusterName, new ArrayList<>());
    }
	// 遍歷最新的實(shí)例集合數(shù)據(jù)
    for (Instance instance : instances) {
        try {
            if (instance == null) {
                Loggers.SRV_LOG.error("[NACOS-DOM] received malformed ip: null");
                continue;
            }
            // 如果集群名稱為null ,則設(shè)置默認(rèn)的集群名稱 DEFAULT
            if (StringUtils.isEmpty(instance.getClusterName())) {
                instance.setClusterName(UtilsAndCommons.DEFAULT_CLUSTER_NAME);
            }
            // 如果當(dāng)前 service 的clusterMap不包括 實(shí)例的 集群名稱,則需要?jiǎng)?chuàng)建新的集群對(duì)象
            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);
            }

            // 如果當(dāng)前 ipMap 不包括 當(dāng)前實(shí)例的 集群名稱,則需要?jiǎng)?chuàng)建新的集群對(duì)象
            List<Instance> clusterIPs = ipMap.get(instance.getClusterName());
            if (clusterIPs == null) {
                clusterIPs = new LinkedList<>();
                ipMap.put(instance.getClusterName(), clusterIPs);
            }
			// 給當(dāng)前的 集群對(duì)象賦值 實(shí)例數(shù)據(jù)。
            clusterIPs.add(instance);
        } catch (Exception e) {
            Loggers.SRV_LOG.error("[NACOS-DOM] failed to process ip: " + instance, e);
        }
    }
	// 遍歷 ipMap對(duì)象,給 clusterMap 替換最新的 entryIPs
    for (Map.Entry<String, List<Instance>> entry : ipMap.entrySet()) {
        //make every ip mine
        List<Instance> entryIPs = entry.getValue();
        // 給 clusterMap 替換最新的 entryIPs
        clusterMap.get(entry.getKey()).updateIps(entryIPs, ephemeral);
    }

    setLastModifiedMillis(System.currentTimeMillis());
    // 發(fā)布
    getPushService().serviceChanged(this);
    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());

}

以上代碼先根據(jù)當(dāng)前服務(wù)下的集群信息構(gòu)造構(gòu)造ipMap對(duì)象,然后遍歷最新的實(shí)例集合數(shù)據(jù)更新ipMap對(duì)象,最后循環(huán)調(diào)用clusterMap.get(entry.getKey()).updateIps(entryIPs, ephemeral)方法來更新當(dāng)前集群中的實(shí)例列表數(shù)據(jù)。updateIps方法代碼如下:

public void updateIps(List<Instance> ips, boolean ephemeral) {
    // 獲取 本集群中的 實(shí)例集合
    Set<Instance> toUpdateInstances = ephemeral ? ephemeralInstances : persistentInstances;
    // 根據(jù)old的實(shí)例數(shù)據(jù) 構(gòu)建 hashmap
    HashMap<String, Instance> oldIpMap = new HashMap<>(toUpdateInstances.size());
    // 根據(jù)實(shí)例的 key 添加到 oldIpMap中
    for (Instance ip : toUpdateInstances) {
        oldIpMap.put(ip.getDatumKey(), ip);
    }
    // 獲取更新的 實(shí)例數(shù)據(jù) List
    List<Instance> updatedIPs = updatedIps(ips, oldIpMap.values());
    if (updatedIPs.size() > 0) {
        for (Instance ip : updatedIPs) {
            Instance oldIP = oldIpMap.get(ip.getDatumKey());

            // do not update the ip validation status of updated ips
            // because the checker has the most precise result
            // Only when ip is not marked, don't we update the health status of IP:
            if (!ip.isMarked()) {
                ip.setHealthy(oldIP.isHealthy());
            }
            if (ip.isHealthy() != oldIP.isHealthy()) {
                // ip validation status updated
                Loggers.EVT_LOG.info("{} {SYNC} IP-{} {}:{}@{}", getService().getName(),
                                     (ip.isHealthy() ? "ENABLED" : "DISABLED"), ip.getIp(), ip.getPort(), getName());
            }

            if (ip.getWeight() != oldIP.getWeight()) {
                // ip validation status updated
                Loggers.EVT_LOG.info("{} {SYNC} {IP-UPDATED} {}->{}", getService().getName(), oldIP.toString(),
                                     ip.toString());
            }
        }
    }
    // 獲取新增的 實(shí)例數(shù)據(jù)
    List<Instance> newIPs = subtract(ips, oldIpMap.values());
    if (newIPs.size() > 0) {
        Loggers.EVT_LOG
            .info("{} {SYNC} {IP-NEW} cluster: {}, new ips size: {}, content: {}", getService().getName(),
                  getName(), newIPs.size(), newIPs.toString());

        for (Instance ip : newIPs) {
            HealthCheckStatus.reset(ip);
        }
    }
    // 獲取刪除的 實(shí)例數(shù)據(jù)
    List<Instance> deadIPs = subtract(oldIpMap.values(), ips);

    if (deadIPs.size() > 0) {
        Loggers.EVT_LOG
            .info("{} {SYNC} {IP-DEAD} cluster: {}, dead ips size: {}, content: {}", getService().getName(),
                  getName(), deadIPs.size(), deadIPs.toString());

        for (Instance ip : deadIPs) {
            HealthCheckStatus.remv(ip);
        }
    }
    // 根據(jù)傳進(jìn)來的 實(shí)例集合 創(chuàng)建需要更新的實(shí)例set 
    toUpdateInstances = new HashSet<>(ips);

    // 直接替換
    if (ephemeral) {
        ephemeralInstances = toUpdateInstances;
    } else {
        persistentInstances = toUpdateInstances;
    }

以上代碼就是更新cluster對(duì)象下的實(shí)例數(shù)據(jù)邏輯,根據(jù)代碼可知在cluster對(duì)象中更新實(shí)例數(shù)據(jù)就是拿傳進(jìn)來的實(shí)例列表創(chuàng)建set集合直接替換的。

二、服務(wù)實(shí)例列表拉取

客戶端程序啟動(dòng)之后,會(huì)執(zhí)行com.alibaba.cloud.nacos.discovery.NacosWatch類的start()方法,此方法中會(huì)執(zhí)行以下語句,

namingService.subscribe(properties.getService(), properties.getGroup(),
                        Arrays.asList(properties.getClusterName()), eventListener);

此方法用來獲取當(dāng)前服務(wù)的實(shí)例數(shù)據(jù),subscribe方法代碼如下,

public void subscribe(String serviceName, String groupName, List<String> clusters, EventListener listener)
    throws NacosException {
    // 獲取服務(wù)列表數(shù)據(jù)
    hostReactor.subscribe(NamingUtils.getGroupedName(serviceName, groupName), StringUtils.join(clusters, ","),
                          listener);
}

通過hostReactor.subscribe方法獲取服務(wù)列表數(shù)據(jù),subscribe方法的代碼如下,

public void subscribe(String serviceName, String clusters, EventListener eventListener) {
    notifier.registerListener(serviceName, clusters, eventListener);
    // 獲取服務(wù)列表數(shù)據(jù)
    getServiceInfo(serviceName, clusters);
}

通過getServiceInfo方法獲取服務(wù)列表數(shù)據(jù),getServiceInfo的代碼如下:

NAMING_LOGGER.debug("failover-mode: " + failoverReactor.isFailoverSwitch());
String key = ServiceInfo.getKey(serviceName, clusters);
if (failoverReactor.isFailoverSwitch()) {
    return failoverReactor.getService(key);
}
// 根據(jù)服務(wù)名稱和集群名稱獲取本地的服務(wù)列表數(shù)據(jù)
ServiceInfo serviceObj = getServiceInfo0(serviceName, clusters);
if (null == serviceObj) {
    serviceObj = new ServiceInfo(serviceName, clusters);
    serviceInfoMap.put(serviceObj.getKey(), serviceObj);
    updatingMap.put(serviceName, new Object());
    // 如果本地服務(wù)實(shí)例數(shù)據(jù)為null,則去獲取最新的服務(wù)實(shí)例列表
    updateServiceNow(serviceName, clusters);
    updatingMap.remove(serviceName);

} else if (updatingMap.containsKey(serviceName)) {
    if (UPDATE_HOLD_INTERVAL > 0) {
        // hold a moment waiting for update finish
        synchronized (serviceObj) {
            try {
                serviceObj.wait(UPDATE_HOLD_INTERVAL);
            } catch (InterruptedException e) {
                NAMING_LOGGER
                    .error("[getServiceInfo] serviceName:" + serviceName + ", clusters:" + clusters, e);
            }
        }
    }
}
scheduleUpdateIfAbsent(serviceName, clusters);
return serviceInfoMap.get(serviceObj.getKey());

以上代碼可知,會(huì)根據(jù)服務(wù)名稱和clusters名稱獲取本地緩存serviceInfoMap對(duì)象中的服務(wù)列表數(shù)據(jù)。如果本地服務(wù)實(shí)例數(shù)據(jù)為null,則通過updateServiceNow方法去nacos服務(wù)端獲取最新的服務(wù)實(shí)例列表。updateServiceNow方法代碼如下:

try {
    // 更新本地服務(wù)方法
    updateService(serviceName, clusters);
} catch (NacosException e) {
    NAMING_LOGGER.error("[NA] failed to update serviceName: " + serviceName, e);
}

updateService的代碼如下:

public void updateService(String serviceName, String clusters) throws NacosException {
    ServiceInfo oldService = getServiceInfo0(serviceName, clusters);
    try {
		// 調(diào)用服務(wù)代理類獲取服務(wù)實(shí)例列表,pushReceiver.getUdpPort()會(huì)隨機(jī)生成一個(gè)udp端口
        String result = serverProxy.queryList(serviceName, clusters, pushReceiver.getUdpPort(), false);
        if (StringUtils.isNotEmpty(result)) {
            // 如果 result不為空,則向本地緩存 serviceInfoMap 添加服務(wù)實(shí)例列表
            processServiceJson(result);
        }
    } finally {
        if (oldService != null) {
            synchronized (oldService) {
                oldService.notifyAll();
            }
        }
    }
}

通過調(diào)用服務(wù)代理類serverProxy的queryList方法獲取服務(wù)實(shí)例列表,pushReceiver.getUdpPort()會(huì)獲pushReceiver的udp端口,pushReceiver對(duì)象是一個(gè)udp數(shù)據(jù)接收類,用來接收nacos服務(wù)器發(fā)送的udp數(shù)據(jù),比如服務(wù)實(shí)例更新的消息。serverProxy.query方法的代碼如下:

public String queryList(String serviceName, String clusters, int udpPort, boolean healthyOnly)
    throws NacosException {
	// 構(gòu)造請(qǐng)求參數(shù)
    final Map<String, String> params = new HashMap<String, String>(8);
    params.put(CommonParams.NAMESPACE_ID, namespaceId);
    params.put(CommonParams.SERVICE_NAME, serviceName);
    params.put("clusters", clusters);
    // 客戶端的upd端口,服務(wù)端回調(diào)客戶端udp端口會(huì)用到
    params.put("udpPort", String.valueOf(udpPort));
    params.put("clientIP", NetUtils.localIP());
    params.put("healthyOnly", String.valueOf(healthyOnly));
	// 向nacos服務(wù)器獲取服務(wù)列表數(shù)據(jù),并返回
    return reqApi(UtilAndComs.nacosUrlBase + "/instance/list", params, HttpMethod.GET);
}

在構(gòu)造的請(qǐng)求參數(shù)中包括了客戶端的udpPort,該參數(shù)在服務(wù)端回調(diào)接口會(huì)用到。reqApi方法其實(shí)就向nacos服務(wù)器的/nacos/v1/ns/instance/list接口發(fā)送了請(qǐng)求消息,該接口對(duì)應(yīng)的nacos服務(wù)端的源碼的naming工程中InstanceController的list方法,代碼如下,

@GetMapping("/list")
@Secured(parser = NamingResourceParser.class, action = ActionTypes.READ)
public ObjectNode list(HttpServletRequest request) throws Exception {

    String namespaceId = WebUtils.optional(request, CommonParams.NAMESPACE_ID, Constants.DEFAULT_NAMESPACE_ID);
    String serviceName = WebUtils.required(request, CommonParams.SERVICE_NAME);
    NamingUtils.checkServiceNameFormat(serviceName);

    String agent = WebUtils.getUserAgent(request);
    String clusters = WebUtils.optional(request, "clusters", StringUtils.EMPTY);
    String clientIP = WebUtils.optional(request, "clientIP", StringUtils.EMPTY);
    int udpPort = Integer.parseInt(WebUtils.optional(request, "udpPort", "0"));
    String env = WebUtils.optional(request, "env", StringUtils.EMPTY);
    boolean isCheck = Boolean.parseBoolean(WebUtils.optional(request, "isCheck", "false"));

    String app = WebUtils.optional(request, "app", StringUtils.EMPTY);

    String tenant = WebUtils.optional(request, "tid", StringUtils.EMPTY);

    boolean healthyOnly = Boolean.parseBoolean(WebUtils.optional(request, "healthyOnly", "false"));
    // 獲取實(shí)例列表數(shù)據(jù)
    return doSrvIpxt(namespaceId, serviceName, agent, clusters, clientIP, udpPort, env, isCheck, app, tenant,
                     healthyOnly);
}

以上代碼先構(gòu)造相關(guān)參數(shù)信息,然后通過doSrvIpxt方法來獲取實(shí)例列表數(shù)據(jù),doSrvIpxt代碼如下:

public ObjectNode doSrvIpxt(String namespaceId, String serviceName, String agent, String clusters, String clientIP,
                            int udpPort, String env, boolean isCheck, String app, String tid, boolean healthyOnly) throws Exception {

    ClientInfo clientInfo = new ClientInfo(agent);
    ObjectNode result = JacksonUtils.createEmptyJsonNode();
    // 根據(jù)命名空間id和服務(wù)名稱獲取服務(wù)
    Service service = serviceManager.getService(namespaceId, serviceName);
    long cacheMillis = switchDomain.getDefaultCacheMillis();

    // now try to enable the push
    try {
        // 如果端口大于0 ,并且是支持的客戶端
        if (udpPort > 0 && pushService.canEnablePush(agent)) {
            // 添加 PushClient 對(duì)象
            pushService
                .addClient(namespaceId, serviceName, clusters, agent, new InetSocketAddress(clientIP, udpPort),
                           pushDataSource, tid, app);
            cacheMillis = switchDomain.getPushCacheMillis(serviceName);
        }
    } catch (Exception e) {
        Loggers.SRV_LOG
            .error("[NACOS-API] failed to added push client {}, {}:{}", clientInfo, clientIP, udpPort, e);
        cacheMillis = switchDomain.getDefaultCacheMillis();
    }
    // 如果服務(wù)對(duì)象為 null ,則構(gòu)造數(shù)據(jù)返回
    if (service == null) {
        if (Loggers.SRV_LOG.isDebugEnabled()) {
            Loggers.SRV_LOG.debug("no instance to serve for service: {}", serviceName);
        }
        result.put("name", serviceName);
        result.put("clusters", clusters);
        result.put("cacheMillis", cacheMillis);
        result.replace("hosts", JacksonUtils.createEmptyArrayNode());
        return result;
    }
    // 檢查服務(wù)是否可用
    checkIfDisabled(service);

    List<Instance> srvedIPs;
    // 根據(jù)集群列表獲取具體服務(wù)下面的實(shí)例列表
    srvedIPs = service.srvIPs(Arrays.asList(StringUtils.split(clusters, ",")));

    // filter ips using selector:
    if (service.getSelector() != null && StringUtils.isNotBlank(clientIP)) {
        srvedIPs = service.getSelector().select(clientIP, srvedIPs);
    }
    // 如果實(shí)例數(shù)據(jù)為空,則構(gòu)造數(shù)據(jù)返回
    if (CollectionUtils.isEmpty(srvedIPs)) {

        if (Loggers.SRV_LOG.isDebugEnabled()) {
            Loggers.SRV_LOG.debug("no instance to serve for service: {}", serviceName);
        }

        if (clientInfo.type == ClientInfo.ClientType.JAVA
            && clientInfo.version.compareTo(VersionUtil.parseVersion("1.0.0")) >= 0) {
            result.put("dom", serviceName);
        } else {
            result.put("dom", NamingUtils.getServiceName(serviceName));
        }

        result.put("name", serviceName);
        result.put("cacheMillis", cacheMillis);
        result.put("lastRefTime", System.currentTimeMillis());
        result.put("checksum", service.getChecksum());
        result.put("useSpecifiedURL", false);
        result.put("clusters", clusters);
        result.put("env", env);
        result.set("hosts", JacksonUtils.createEmptyArrayNode());
        result.set("metadata", JacksonUtils.transferToJsonNode(service.getMetadata()));
        return result;
    }

    Map<Boolean, List<Instance>> ipMap = new HashMap<>(2);
    ipMap.put(Boolean.TRUE, new ArrayList<>());
    ipMap.put(Boolean.FALSE, new ArrayList<>());
    // 構(gòu)造健康和不健康的實(shí)例數(shù)據(jù)
    for (Instance ip : srvedIPs) {
        ipMap.get(ip.isHealthy()).add(ip);
    }

    if (isCheck) {
        result.put("reachProtectThreshold", false);
    }

    double threshold = service.getProtectThreshold();

    if ((float) ipMap.get(Boolean.TRUE).size() / srvedIPs.size() <= threshold) {

        Loggers.SRV_LOG.warn("protect threshold reached, return all ips, service: {}", serviceName);
        if (isCheck) {
            result.put("reachProtectThreshold", true);
        }

        ipMap.get(Boolean.TRUE).addAll(ipMap.get(Boolean.FALSE));
        ipMap.get(Boolean.FALSE).clear();
    }

    if (isCheck) {
        result.put("protectThreshold", service.getProtectThreshold());
        result.put("reachLocalSiteCallThreshold", false);

        return JacksonUtils.createEmptyJsonNode();
    }

    ArrayNode hosts = JacksonUtils.createEmptyArrayNode();
    // 構(gòu)造返回的實(shí)例列表對(duì)象
    for (Map.Entry<Boolean, List<Instance>> entry : ipMap.entrySet()) {
        List<Instance> ips = entry.getValue();

        if (healthyOnly && !entry.getKey()) {
            continue;
        }

        for (Instance instance : ips) {

            // remove disabled instance:
            if (!instance.isEnabled()) {
                continue;
            }

            ObjectNode ipObj = JacksonUtils.createEmptyJsonNode();

            ipObj.put("ip", instance.getIp());
            ipObj.put("port", instance.getPort());
            // deprecated since nacos 1.0.0:
            ipObj.put("valid", entry.getKey());
            ipObj.put("healthy", entry.getKey());
            ipObj.put("marked", instance.isMarked());
            ipObj.put("instanceId", instance.getInstanceId());
            ipObj.set("metadata", JacksonUtils.transferToJsonNode(instance.getMetadata()));
            ipObj.put("enabled", instance.isEnabled());
            ipObj.put("weight", instance.getWeight());
            ipObj.put("clusterName", instance.getClusterName());
            if (clientInfo.type == ClientInfo.ClientType.JAVA
                && clientInfo.version.compareTo(VersionUtil.parseVersion("1.0.0")) >= 0) {
                ipObj.put("serviceName", instance.getServiceName());
            } else {
                ipObj.put("serviceName", NamingUtils.getServiceName(instance.getServiceName()));
            }

            ipObj.put("ephemeral", instance.isEphemeral());
            hosts.add(ipObj);

        }
    }

    result.replace("hosts", hosts);
    if (clientInfo.type == ClientInfo.ClientType.JAVA
        && clientInfo.version.compareTo(VersionUtil.parseVersion("1.0.0")) >= 0) {
        result.put("dom", serviceName);
    } else {
        result.put("dom", NamingUtils.getServiceName(serviceName));
    }
    result.put("name", serviceName);
    result.put("cacheMillis", cacheMillis);
    result.put("lastRefTime", System.currentTimeMillis());
    result.put("checksum", service.getChecksum());
    result.put("useSpecifiedURL", false);
    result.put("clusters", clusters);
    result.put("env", env);
    result.replace("metadata", JacksonUtils.transferToJsonNode(service.getMetadata()));
    return result;
}

以上代碼其實(shí)就是根據(jù)命名空間id和服務(wù)名稱獲取服務(wù)對(duì)象,然后根據(jù)不同情況構(gòu)造返回對(duì)象,正常情況會(huì)構(gòu)造一個(gè)ServiceInfo類型的ObjectNode對(duì)象,整個(gè)具體過程請(qǐng)看上面的代碼注釋。最后返回構(gòu)造的對(duì)象。

客戶端中拿到請(qǐng)求/nacos/v1/ns/instance/list接口的返回值之后會(huì)轉(zhuǎn)成一個(gè)ServiceInfo對(duì)象,并且把該對(duì)象賦值給本地的緩存對(duì)象serviceInfoMap,processServiceJson關(guān)鍵代碼如下:

// 將返回值轉(zhuǎn)換成 ServiceInfo 類型的對(duì)象
ServiceInfo serviceInfo = JacksonUtils.toObj(json, ServiceInfo.class);
// 把該對(duì)象添加到本地緩存中
serviceInfoMap.put(serviceInfo.getKey(), serviceInfo);

三、定時(shí)心跳任務(wù)

在客戶端向nacos服務(wù)端注冊(cè)服務(wù)的過程中,會(huì)調(diào)用com.alibaba.nacos.client.naming.NacosNamingService#registerInstance(java.lang.String, java.lang.String, com.alibaba.nacos.api.naming.pojo.Instance)方法,在該代碼中有個(gè)判斷邏輯,如果是臨時(shí)實(shí)例則會(huì)創(chuàng)建一個(gè)BeatInfo對(duì)象添加到beatReactor中。代碼如下:

public void registerInstance(String serviceName, String groupName, Instance instance) throws NacosException {
    // 檢查 實(shí)例是否合法
    // heart beat timeout must(默認(rèn)15秒) > heart beat interval (默認(rèn)5秒)
    // ip delete timeout must(默認(rèn)30 秒)  > heart beat interval
    NamingUtils.checkInstanceIsLegal(instance);
    // 構(gòu)建 groupName@@serviceName
    String groupedServiceName = NamingUtils.getGroupedName(serviceName, groupName);
    // 如果是臨時(shí)實(shí)例,則創(chuàng)建心跳信息,定時(shí)給nacos服務(wù)發(fā)送
    if (instance.isEphemeral()) {
        // 構(gòu)造心跳信息
        BeatInfo beatInfo = this.beatReactor.buildBeatInfo(groupedServiceName, instance);
        // 執(zhí)行心跳定時(shí)任務(wù)
        this.beatReactor.addBeatInfo(groupedServiceName, beatInfo);
    }
	// 向 nacos-service 注冊(cè)實(shí)例
    this.serverProxy.registerService(groupedServiceName, groupName, instance);
}

beatInfo對(duì)象用來存儲(chǔ)心跳信息,buildBeatInfo方法代碼如下,

public BeatInfo buildBeatInfo(String groupedServiceName, Instance instance) {
    BeatInfo beatInfo = new BeatInfo();
    beatInfo.setServiceName(groupedServiceName);
    beatInfo.setIp(instance.getIp());
    beatInfo.setPort(instance.getPort());
    beatInfo.setCluster(instance.getClusterName());
    beatInfo.setWeight(instance.getWeight());
    beatInfo.setMetadata(instance.getMetadata());
    beatInfo.setScheduled(false);
    beatInfo.setPeriod(instance.getInstanceHeartBeatInterval());
    return beatInfo;
}

beatReactor中有一個(gè)ScheduledExecutorService類型的executorService實(shí)例用來執(zhí)行定時(shí)的線程,addBeatInfo的代碼如下,

public void addBeatInfo(String serviceName, BeatInfo beatInfo) {
    NAMING_LOGGER.info("[BEAT] adding beat: {} to beat map.", beatInfo);
    String key = buildKey(serviceName, beatInfo.getIp(), beatInfo.getPort());
    BeatInfo existBeat = null;
    //fix #1733
    if ((existBeat = dom2Beat.remove(key)) != null) {
        existBeat.setStopped(true);
    }
    dom2Beat.put(key, beatInfo);
    // 線程池添加定時(shí)任務(wù),默認(rèn) 5 秒鐘之后 執(zhí)行 BeatTask
    executorService.schedule(new BeatTask(beatInfo), beatInfo.getPeriod(), TimeUnit.MILLISECONDS);
    MetricsMonitor.getDom2BeatSizeMonitor().set(dom2Beat.size());
}

根據(jù)上面的executorService.schedule()代碼可知,BeatTask線程在固定的秒數(shù)之后執(zhí)行,而BeatTask實(shí)現(xiàn)了Runnable接口,即執(zhí)行BeatTask的run方法 。BeatTask的run方法代碼如下,

public void run() {
    // 如果 beatInfo 設(shè)置了 stop ,則停止
    if (beatInfo.isStopped()) {
        return;
    }
    // 獲取下一次延期執(zhí)行的時(shí)間
    long nextTime = beatInfo.getPeriod();
    try {
        // 向服務(wù)端發(fā)送心跳信息
        JsonNode result = serverProxy.sendBeat(beatInfo, BeatReactor.this.lightBeatEnabled);
        long interval = result.get("clientBeatInterval").asLong();
        boolean lightBeatEnabled = false;
        if (result.has(CommonParams.LIGHT_BEAT_ENABLED)) {
            lightBeatEnabled = result.get(CommonParams.LIGHT_BEAT_ENABLED).asBoolean();
        }
        BeatReactor.this.lightBeatEnabled = lightBeatEnabled;
        if (interval > 0) {
            nextTime = interval;
        }
        int code = NamingResponseCode.OK;
        if (result.has(CommonParams.CODE)) {
            code = result.get(CommonParams.CODE).asInt();
        }
        if (code == NamingResponseCode.RESOURCE_NOT_FOUND) {
            Instance instance = new Instance();
            instance.setPort(beatInfo.getPort());
            instance.setIp(beatInfo.getIp());
            instance.setWeight(beatInfo.getWeight());
            instance.setMetadata(beatInfo.getMetadata());
            instance.setClusterName(beatInfo.getCluster());
            instance.setServiceName(beatInfo.getServiceName());
            instance.setInstanceId(instance.getInstanceId());
            instance.setEphemeral(true);
            try {
                serverProxy.registerService(beatInfo.getServiceName(),
                                            NamingUtils.getGroupName(beatInfo.getServiceName()), instance);
            } catch (Exception ignore) {
            }
        }
    } catch (NacosException ex) {
        NAMING_LOGGER.error("[CLIENT-BEAT] failed to send beat: {}, code: {}, msg: {}",
                            JacksonUtils.toJson(beatInfo), ex.getErrCode(), ex.getErrMsg());

    }
    // 重新提交定時(shí)任務(wù),延期發(fā)送心跳信息
    executorService.schedule(new BeatTask(beatInfo), nextTime, TimeUnit.MILLISECONDS);
}

以上代碼先獲取下一次延期執(zhí)行的時(shí)間,再通過serverProxy.sendBeat()向服務(wù)端發(fā)送心跳信息,最后重新提交定時(shí)任務(wù),延期發(fā)送心跳信息,serverProxy.sendBeat()代碼如下,

public JsonNode sendBeat(BeatInfo beatInfo, boolean lightBeatEnabled) throws NacosException {
    if (NAMING_LOGGER.isDebugEnabled()) {
        NAMING_LOGGER.debug("[BEAT] {} sending beat to server: {}", namespaceId, beatInfo.toString());
    }
    Map<String, String> params = new HashMap<String, String>(8);
    Map<String, String> bodyMap = new HashMap<String, String>(2);
    if (!lightBeatEnabled) {
        bodyMap.put("beat", JacksonUtils.toJson(beatInfo));
    }
    params.put(CommonParams.NAMESPACE_ID, namespaceId);
    params.put(CommonParams.SERVICE_NAME, beatInfo.getServiceName());
    params.put(CommonParams.CLUSTER_NAME, beatInfo.getCluster());
    params.put("ip", beatInfo.getIp());
    params.put("port", String.valueOf(beatInfo.getPort()));
    // 向nacos服務(wù)器發(fā)送心跳數(shù)據(jù),并返回
    String result = reqApi(UtilAndComs.nacosUrlBase + "/instance/beat", params, bodyMap, HttpMethod.PUT);
    return JacksonUtils.toObj(result);
}

reqApi方法其實(shí)就向nacos服務(wù)器端的/nacos/v1/ns/instance/beat接口發(fā)送了put類型的請(qǐng)求消息,該接口對(duì)應(yīng)的nacos服務(wù)端的源碼的naming工程中InstanceController的beat方法,beat方法的代碼如下,

@CanDistro
@PutMapping("/beat")
@Secured(parser = NamingResourceParser.class, action = ActionTypes.WRITE)
public ObjectNode beat(HttpServletRequest request) throws Exception {
    ObjectNode result = JacksonUtils.createEmptyJsonNode();
    result.put(SwitchEntry.CLIENT_BEAT_INTERVAL, switchDomain.getClientBeatInterval());
    // 獲取心跳信息
    String beat = WebUtils.optional(request, "beat", StringUtils.EMPTY);
    RsInfo clientBeat = null;
    // 如果 beat 數(shù)據(jù)不為空,則構(gòu)造 RsInfo 類型的  clientBeat 實(shí)例
    if (StringUtils.isNotBlank(beat)) {
        clientBeat = JacksonUtils.toObj(beat, RsInfo.class);
    }
    // 獲取集群名稱
    String clusterName = WebUtils
        .optional(request, CommonParams.CLUSTER_NAME, UtilsAndCommons.DEFAULT_CLUSTER_NAME);
    // 獲取 實(shí)例的 ip
    String ip = WebUtils.optional(request, "ip", StringUtils.EMPTY);
    // 獲取 實(shí)例的 端口
    int port = Integer.parseInt(WebUtils.optional(request, "port", "0"));
    // 如果 clientBeat 不為空,則設(shè)置 相關(guān)的信息
    if (clientBeat != null) {
        if (StringUtils.isNotBlank(clientBeat.getCluster())) {
            clusterName = clientBeat.getCluster();
        } else {
            // fix #2533
            clientBeat.setCluster(clusterName);
        }
        ip = clientBeat.getIp();
        port = clientBeat.getPort();
    }
    // 獲取 namespaceId
    String namespaceId = WebUtils.optional(request, CommonParams.NAMESPACE_ID, Constants.DEFAULT_NAMESPACE_ID);
    // 獲取 serviceName
    String serviceName = WebUtils.required(request, CommonParams.SERVICE_NAME);
    // 檢查 ServiceName 的格式
    NamingUtils.checkServiceNameFormat(serviceName);
    Loggers.SRV_LOG.debug("[CLIENT-BEAT] full arguments: beat: {}, serviceName: {}", clientBeat, serviceName);
    // 根據(jù) 參數(shù) 獲取 具體的實(shí)例
    Instance instance = serviceManager.getInstance(namespaceId, serviceName, clusterName, ip, port);
    // 如果 實(shí)例為 空
    if (instance == null) {
        // 如果 clientBeat 為空 則構(gòu)造參數(shù) code 為 20404的結(jié)果返回
        if (clientBeat == null) {
            result.put(CommonParams.CODE, NamingResponseCode.RESOURCE_NOT_FOUND);
            return result;
        }

        Loggers.SRV_LOG.warn("[CLIENT-BEAT] The instance has been removed for health mechanism, "
                             + "perform data compensation operations, beat: {}, serviceName: {}", clientBeat, serviceName);
        // 如果 clientBeat 不為空 則構(gòu)造 instance 數(shù)據(jù),向 serviceManager 注冊(cè)實(shí)例。
        instance = new Instance();
        instance.setPort(clientBeat.getPort());
        instance.setIp(clientBeat.getIp());
        instance.setWeight(clientBeat.getWeight());
        instance.setMetadata(clientBeat.getMetadata());
        instance.setClusterName(clusterName);
        instance.setServiceName(serviceName);
        instance.setInstanceId(instance.getInstanceId());
        instance.setEphemeral(clientBeat.isEphemeral());

        serviceManager.registerInstance(namespaceId, serviceName, instance);
    }
    // 根據(jù)服務(wù)名稱獲取 服務(wù)
    Service service = serviceManager.getService(namespaceId, serviceName);
    // 如果服務(wù)為空 ,則拋異常
    if (service == null) {
        throw new NacosException(NacosException.SERVER_ERROR,
                                 "service not found: " + serviceName + "@" + namespaceId);
    }
    // 如果 clientBeat 為空,則創(chuàng)建該對(duì)象
    if (clientBeat == null) {
        clientBeat = new RsInfo();
        clientBeat.setIp(ip);
        clientBeat.setPort(port);
        clientBeat.setCluster(clusterName);
    }
    // 處理客戶端的 心跳對(duì)象
    service.processClientBeat(clientBeat);
    //
    result.put(CommonParams.CODE, NamingResponseCode.OK);
    if (instance.containsMetadata(PreservedMetadataKeys.HEART_BEAT_INTERVAL)) {
        result.put(SwitchEntry.CLIENT_BEAT_INTERVAL, instance.getInstanceHeartBeatInterval());
    }
    result.put(SwitchEntry.LIGHT_BEAT_ENABLED, switchDomain.isLightBeatEnabled());
    return result;
}

先獲取心跳信息,然后構(gòu)造RsInfo類型的clientBeat實(shí)例。然后通過service.processClientBeat(clientBeat)方法處理客戶端的心跳對(duì)象,processClientBeat方法的代碼如下,

public void processClientBeat(final RsInfo rsInfo) {
    // 構(gòu)造 ClientBeatProcessor 對(duì)象
    ClientBeatProcessor clientBeatProcessor = new ClientBeatProcessor();
    clientBeatProcessor.setService(this);
    clientBeatProcessor.setRsInfo(rsInfo);
    // 定時(shí)執(zhí)行 ClientBeatProcessor 對(duì)象,這里是立即執(zhí)行,延期時(shí)間為 0
    HealthCheckReactor.scheduleNow(clientBeatProcessor);
}

ClientBeatProcessor是一個(gè)實(shí)現(xiàn)了Runnable的類,HealthCheckReactor是一個(gè)定時(shí)任務(wù)線程池,scheduleNow方法表示立即執(zhí)行clientBeatProcessor對(duì)象的run方法,clientBeatProcessor.run方法代碼如下,

public void run() {
    Service service = this.service;
    if (Loggers.EVT_LOG.isDebugEnabled()) {
        Loggers.EVT_LOG.debug("[CLIENT-BEAT] processing beat: {}", rsInfo.toString());
    }
    // 獲取ip
    String ip = rsInfo.getIp();
    // 獲取 集群名稱
    String clusterName = rsInfo.getCluster();
    // 獲取端口
    int port = rsInfo.getPort();
    // 從服務(wù)對(duì)象中獲取集群對(duì)象
    Cluster cluster = service.getClusterMap().get(clusterName);
    // 從集群對(duì)象中獲取所有的臨時(shí)實(shí)例列表
    List<Instance> instances = cluster.allIPs(true);

    for (Instance instance : instances) {
        // 找到 ip 和端口相同的 實(shí)例數(shù)據(jù)
        if (instance.getIp().equals(ip) && instance.getPort() == port) {
            if (Loggers.EVT_LOG.isDebugEnabled()) {
                Loggers.EVT_LOG.debug("[CLIENT-BEAT] refresh beat: {}", rsInfo.toString());
            }
            // 更新 最后心跳時(shí)間
            instance.setLastBeat(System.currentTimeMillis());
            if (!instance.isMarked() && !instance.isHealthy()) {
                instance.setHealthy(true);
                Loggers.EVT_LOG
                    .info("service: {} {POS} {IP-ENABLED} valid: {}:{}@{}, region: {}, msg: client beat ok",
                          cluster.getService().getName(), ip, port, cluster.getName(),
                          UtilsAndCommons.LOCALHOST_SITE);
                getPushService().serviceChanged(service);
            }
        }
    }
}

以上代碼可知,該方法主要用來更新客戶端實(shí)例的最后心跳時(shí)間。

三、服務(wù)端接口

一、定時(shí)檢查服務(wù)實(shí)例任務(wù)

在客戶端注冊(cè)服務(wù)的時(shí)候,會(huì)調(diào)用nacos服務(wù)端的com.alibaba.nacos.naming.controllers.InstanceController#register方法,其中會(huì)調(diào)用createEmptyService方法用來創(chuàng)建空的服務(wù)對(duì)象,最后會(huì)調(diào)用service.init()方法用來初始化服務(wù)對(duì)象,init方法代碼如下

public void init() {
    // 定時(shí)執(zhí)行 service 的 run 方法 處理超時(shí)的 instance
    HealthCheckReactor.scheduleCheck(clientBeatCheckTask);
    for (Map.Entry<String, Cluster> entry : clusterMap.entrySet()) {
        entry.getValue().setService(this);
        entry.getValue().init();
    }
}

通過調(diào)用HealthCheckReactor.scheduleCheck()方法來定時(shí)執(zhí)行clientBeatCheckTask,scheduleCheck的代碼如下,

public static void scheduleCheck(ClientBeatCheckTask task) {
    // 5秒之后執(zhí)行 task,并且每次執(zhí)行task完之后,5秒之后再次執(zhí)行 task
    futureMap.computeIfAbsent(task.taskKey(),
                              k -> GlobalExecutor.scheduleNamingHealth(task, 5000, 5000, TimeUnit.MILLISECONDS));
}

以上代碼給定時(shí)任務(wù)線程池GlobalExecutor提交了一個(gè)task任務(wù),其中task是一個(gè)實(shí)現(xiàn)了Runable接口的類,線程池每次執(zhí)行的就是ClientBeatCheckTask 的run方法,run方法代碼如下,

public void run() {
    try {
        if (!getDistroMapper().responsible(service.getName())) {
            return;
        }

        if (!getSwitchDomain().isHealthCheckEnabled()) {
            return;
        }
        // 獲取該服務(wù)下面的所有 注冊(cè)實(shí)例集合
        List<Instance> instances = service.allIPs(true);
        // first set health status of instances:
        for (Instance instance : instances) {
            // 如果 當(dāng)前時(shí)間 減去 實(shí)例的最新心跳時(shí)間 如果大于 實(shí)例配置的心跳超時(shí)時(shí)間(默認(rèn)15秒)
            // 并且 實(shí)例的健康狀態(tài) true
            // 則設(shè)置服務(wù)的健康狀態(tài)為 false
            if (System.currentTimeMillis() - instance.getLastBeat() > instance.getInstanceHeartBeatTimeOut()) {
                if (!instance.isMarked()) {
                    if (instance.isHealthy()) {
                        instance.setHealthy(false);
                        Loggers.EVT_LOG
                            .info("{POS} {IP-DISABLED} valid: {}:{}@{}@{}, region: {}, msg: client timeout after {}, last beat: {}",
                                  instance.getIp(), instance.getPort(), instance.getClusterName(),
                                  service.getName(), UtilsAndCommons.LOCALHOST_SITE,
                                  instance.getInstanceHeartBeatTimeOut(), instance.getLastBeat());
                        getPushService().serviceChanged(service);
                        ApplicationUtils.publishEvent(new InstanceHeartbeatTimeoutEvent(this, instance));
                    }
                }
            }
        }
        if (!getGlobalConfig().isExpireInstance()) {
            return;
        }
        // then remove obsolete instances:
        for (Instance instance : instances) {

            if (instance.isMarked()) {
                continue;
            }
            // 如果 當(dāng)前時(shí)間 減去 實(shí)例的最新心跳時(shí)間 如果大于 實(shí)例配置的刪除超時(shí)時(shí)間(默認(rèn)30秒)
            // 則會(huì)調(diào)用 deleteIp 刪除方法刪除實(shí)例
            if (System.currentTimeMillis() - instance.getLastBeat() > instance.getIpDeleteTimeout()) {
                // delete instance
                Loggers.SRV_LOG.info("[AUTO-DELETE-IP] service: {}, ip: {}", service.getName(),
                                     JacksonUtils.toJson(instance));
                // 刪除實(shí)例
                deleteIp(instance);
            }
        }

    } catch (Exception e) {
        Loggers.SRV_LOG.warn("Exception while processing client beat time out.", e);
    }

}

以上代碼就兩個(gè)邏輯,一個(gè)邏輯是判斷當(dāng)前時(shí)間減去實(shí)例的最新心跳時(shí)間是否大于實(shí)例配置的心跳超時(shí)時(shí)間(默認(rèn)15秒),如果大于則設(shè)置實(shí)例的健康狀態(tài)為false;第二個(gè)邏輯是 判斷當(dāng)前時(shí)間減去實(shí)例的最新心跳時(shí)間 是否大于實(shí)例配置的刪除超時(shí)時(shí)間(默認(rèn)30秒),如果大于則調(diào)用deleteIp(instance);刪除該實(shí)例,deleteIp的代碼如下,

NamingProxy.Request request = NamingProxy.Request.newRequest();
request.appendParam("ip", instance.getIp()).appendParam("port", String.valueOf(instance.getPort()))
    .appendParam("ephemeral", "true").appendParam("clusterName", instance.getClusterName())
    .appendParam("serviceName", service.getName()).appendParam("namespaceId", service.getNamespaceId());
// 構(gòu)造url地址
String url = "http://" + IPUtil.localHostIP() + IPUtil.IP_PORT_SPLITER + EnvUtil.getPort() + EnvUtil.getContextPath()
    + UtilsAndCommons.NACOS_NAMING_CONTEXT + "/instance?" + request.toUrl();

// delete instance asynchronously:
// 向本地服務(wù)器地址發(fā)送刪除請(qǐng)求
HttpClient.asyncHttpDelete(url, null, null, new Callback<String>() {
    @Override
    public void onReceive(RestResult<String> result) {
        if (!result.ok()) {
            Loggers.SRV_LOG
                .error("[IP-DEAD] failed to delete ip automatically, ip: {}, caused {}, resp code: {}",
                       instance.toJson(), result.getMessage(), result.getCode());
        }
    }
    @Override
    public void onError(Throwable throwable) {
        Loggers.SRV_LOG
            .error("[IP-DEAD] failed to delete ip automatically, ip: {}, error: ", instance.toJson(),
                   throwable);
    }
    @Override
    public void onCancel() {

    }
});

HttpClient.asyncHttpDelete方法其實(shí)就是向 ${spring.cloud.nacos.discovery.server-addr}/nacos/v1/ns/instance 發(fā)送Delete請(qǐng)求。該請(qǐng)求地址對(duì)應(yīng)的nacos服務(wù)端的源碼的naming工程中InstanceController的deregister方法,deregister代碼如下,

@CanDistro
@DeleteMapping
@Secured(parser = NamingResourceParser.class, action = ActionTypes.WRITE)
public String deregister(HttpServletRequest request) throws Exception {
    // 從請(qǐng)求參數(shù)中構(gòu)造實(shí)例對(duì)象
    Instance instance = getIpAddress(request);
    String namespaceId = WebUtils.optional(request, CommonParams.NAMESPACE_ID, Constants.DEFAULT_NAMESPACE_ID);
    String serviceName = WebUtils.required(request, CommonParams.SERVICE_NAME);
    NamingUtils.checkServiceNameFormat(serviceName);

    Service service = serviceManager.getService(namespaceId, serviceName);
    if (service == null) {
        Loggers.SRV_LOG.warn("remove instance from non-exist service: {}", serviceName);
        return "ok";
    }
	// 刪除實(shí)例數(shù)據(jù)
    serviceManager.removeInstance(namespaceId, serviceName, instance.isEphemeral(), instance);
    return "ok";
}

removeInstance方法是關(guān)鍵,用來刪除實(shí)例數(shù)據(jù),removeInstance代碼如下,

public void removeInstance(String namespaceId, String serviceName, boolean ephemeral, Instance... ips)
    throws NacosException {
    // 先獲取服務(wù)對(duì)象
    Service service = getService(namespaceId, serviceName);
    // 服務(wù)對(duì)象加鎖
    synchronized (service) {
        // 調(diào)用刪除實(shí)例對(duì)象的方法
        removeInstance(namespaceId, serviceName, ephemeral, service, ips);
    }
}

removeInstance方法的代碼如下,

private void removeInstance(String namespaceId, String serviceName, boolean ephemeral, Service service,
                            Instance... ips) throws NacosException {
    // 構(gòu)造 key
    String key = KeyBuilder.buildInstanceListKey(namespaceId, serviceName, ephemeral);
    // 獲取服務(wù)下的實(shí)例集合(服務(wù)已有 減去 需要?jiǎng)h除實(shí)例)
    List<Instance> instanceList = substractIpAddresses(service, ephemeral, ips);

    Instances instances = new Instances();
    instances.setInstanceList(instanceList);
    // 根據(jù)KEY更新服務(wù)的實(shí)例
    consistencyService.put(key, instances);
}

substractIpAddresses方法用來獲取該服務(wù)下已經(jīng)減去需要?jiǎng)h除實(shí)例的實(shí)例數(shù)據(jù),其中調(diào)用的updateIpAddresses方法,action值為 remove。removeInstance方法的整體邏輯為通過updateIpAddresses方法拿到該服務(wù)中去掉刪除實(shí)例之后的實(shí)例集合對(duì)象,并把該實(shí)例集合對(duì)象添加到consistencyService對(duì)象中,consistencyService.put(key, instances)里面的邏輯和客戶端注冊(cè)服務(wù)一樣的邏輯。updateIpAddresses方法和consistencyService.put方法已經(jīng)在客戶端服務(wù)注冊(cè)章節(jié)已經(jīng)講了,這里不再講解。

二、服務(wù)實(shí)例更新推送

在客戶端更新服務(wù)實(shí)例的過程中nacos服務(wù)端會(huì)調(diào)用com.alibaba.nacos.naming.core.Service#updateIPs()方法(客戶端注冊(cè)服務(wù)的過程請(qǐng)看客戶端服務(wù)注冊(cè)章節(jié)),在該方法中會(huì)調(diào)用getPushService().serviceChanged(this)來發(fā)布當(dāng)前服務(wù)的修改事件,即會(huì)發(fā)布一個(gè)事件用來通知已經(jīng)和nacos服務(wù)端通信過的客戶端更新客戶端本地的服務(wù)信息。serviceChanged 的代碼如下,

public void serviceChanged(Service service) {
    // merge some change events to reduce the push frequency:
    if (futureMap
        .containsKey(UtilsAndCommons.assembleFullServiceName(service.getNamespaceId(), service.getName()))) {
        return;
    }
    // 發(fā)布服務(wù)修改事件
    this.applicationContext.publishEvent(new ServiceChangeEvent(this, service));
}

applicationContext.publishEvent會(huì)觸發(fā)一個(gè)ServiceChangeEvent事件,其實(shí)就是觸發(fā)com.alibaba.nacos.naming.push.PushService#onApplicationEvent方法,其中邏輯為先根據(jù)命名空間id和服務(wù)名稱獲取所有的客戶端map對(duì)象,然后遍歷所有客戶端對(duì)象PushClient 構(gòu)造 ackEntry 對(duì)象,最后向具體的客戶端發(fā)送 upd 消息。獲取關(guān)鍵代碼如下,

if (compressData != null) {
    ackEntry = prepareAckEntry(client, compressData, data, lastRefTime);
} else {
    // 構(gòu)造 ackEntry 對(duì)象
    ackEntry = prepareAckEntry(client, prepareHostsQData(client), lastRefTime);
    // 添加緩存
    if (ackEntry != null) {
        cache.put(key, new org.javatuples.Pair<>(ackEntry.origin.getData(), ackEntry.data));
    }
}

Loggers.PUSH.info("serviceName: {} changed, schedule push for: {}, agent: {}, key: {}",
                  client.getServiceName(), client.getAddrStr(), client.getAgent(),
                  (ackEntry == null ? null : ackEntry.key));
// 向具體的客戶端發(fā)送 upd 消息
udpPush(ackEntry);

以上代碼prepareHostsQData的邏輯就是獲取該服務(wù)下客戶端所屬服務(wù)的所有實(shí)例數(shù)據(jù),并且構(gòu)造具體的Map<String,Object>對(duì)象,prepareHostsQData代碼如下,

private static Map<String, Object> prepareHostsData(PushClient client) throws Exception {
    Map<String, Object> cmd = new HashMap<String, Object>(2);
    cmd.put("type", "dom");
    // 獲取客戶端所屬服務(wù)的所有實(shí)例數(shù)據(jù)
    cmd.put("data", client.getDataSource().getData(client));
    return cmd;
}

udpPush(ackEntry)里面封裝了發(fā)送udp消息的關(guān)鍵代碼,ackEntry封裝了udp的數(shù)據(jù)信息。

在客戶端獲取服務(wù)實(shí)例列表的時(shí)候,會(huì)生成一個(gè)PushReceiver對(duì)象,該對(duì)象用來監(jiān)聽和接收nacos服務(wù)端發(fā)送的udp數(shù)據(jù)。該對(duì)象實(shí)現(xiàn)了Runnable接口,并且在構(gòu)造方法中把自己提交給了一個(gè)內(nèi)部屬性的線程池對(duì)象。構(gòu)造方法如下,

public PushReceiver(HostReactor hostReactor) {
    try {
        this.hostReactor = hostReactor;
        this.udpSocket = new DatagramSocket();
        this.executorService = new ScheduledThreadPoolExecutor(1, new ThreadFactory() {
            @Override
            public Thread newThread(Runnable r) {
                Thread thread = new Thread(r);
                thread.setDaemon(true);
                thread.setName("com.alibaba.nacos.naming.push.receiver");
                return thread;
            }
        });

        this.executorService.execute(this);
    } catch (Exception e) {
        NAMING_LOGGER.error("[NA] init udp socket failed", e);
    }
}

根據(jù)以上代碼可知,所以在創(chuàng)建PushReceiver對(duì)象之后會(huì)執(zhí)行run方法,run方法的代碼如下,

public void run() {
    while (!closed) {
        try {
            // byte[] is initialized with 0 full filled by default
            byte[] buffer = new byte[UDP_MSS];
            // 構(gòu)造 DatagramPacket 對(duì)象
            DatagramPacket packet = new DatagramPacket(buffer, buffer.length);
			// 監(jiān)聽upd數(shù)據(jù)
            udpSocket.receive(packet);
			// 構(gòu)造服務(wù)對(duì)象的string 數(shù)據(jù)
            String json = new String(IoUtils.tryDecompress(packet.getData()), UTF_8).trim();
            NAMING_LOGGER.info("received push data: " + json + " from " + packet.getAddress().toString());

            PushPacket pushPacket = JacksonUtils.toObj(json, PushPacket.class);
            String ack;
            if ("dom".equals(pushPacket.type) || "service".equals(pushPacket.type)) {
                // 更新本地緩存 serviceInfoMap 的服務(wù)對(duì)象
                hostReactor.processServiceJson(pushPacket.data);
                // send ack to server
                ack = "{\"type\": \"push-ack\"" + ", \"lastRefTime\":\"" + pushPacket.lastRefTime + "\", \"data\":"
                    + "\"\"}";
            } else if ("dump".equals(pushPacket.type)) {
                // dump data to server
                ack = "{\"type\": \"dump-ack\"" + ", \"lastRefTime\": \"" + pushPacket.lastRefTime + "\", \"data\":"
                    + "\"" + StringUtils.escapeJavaScript(JacksonUtils.toJson(hostReactor.getServiceInfoMap()))
                    + "\"}";
            } else {
                // do nothing send ack only
                ack = "{\"type\": \"unknown-ack\"" + ", \"lastRefTime\":\"" + pushPacket.lastRefTime
                    + "\", \"data\":" + "\"\"}";
            }

            udpSocket.send(new DatagramPacket(ack.getBytes(UTF_8), ack.getBytes(UTF_8).length,
                                              packet.getSocketAddress()));
        } catch (Exception e) {
            if (closed) {
                return;
            }
            NAMING_LOGGER.error("[NA] error while receiving push data", e);
        }
    }
}

以上代碼用while一直監(jiān)聽udpSocket的客戶端upd的端口。當(dāng)接收到從nacos服務(wù)端發(fā)送過來的udp數(shù)據(jù)之后會(huì)接著調(diào)用 hostReactor.processServiceJson()方法來更新客戶端本地的serviceInfoMap 的服務(wù)對(duì)象。

到此這篇關(guān)于nacos注冊(cè)中心單節(jié)點(diǎn)ap架構(gòu)源碼解析的文章就介紹到這了,更多相關(guān)nacos注冊(cè)中心ap架構(gòu)源碼內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論

日韩国产乱码中文字幕| 男人操女人逼逼视频网站| 五十路熟女av天堂| 蜜桃视频入口久久久| 成人免费做爰高潮视频| 黄色的网站在线免费看| 91九色国产porny蝌蚪| 国产三级影院在线观看| 97人妻色免费视频| 午夜久久久久久久精品熟女| 欧美80老妇人性视频| 性感美女诱惑福利视频| 成年人中文字幕在线观看| 日本一本午夜在线播放| 日本高清撒尿pissing| 色秀欧美视频第一页| 97小视频人妻一区二区| 久久久久久9999久久久久| 中文字幕一区二区三区人妻大片| ka0ri在线视频| 精品久久久久久久久久久久人妻 | 视频 国产 精品 熟女 | 中文字幕亚洲中文字幕| 中文字幕,亚洲人妻| 在线观看视频网站麻豆| av老司机精品在线观看| 亚洲欧美成人综合在线观看| 日韩一个色综合导航| 亚洲熟女久久久36d| 人妻少妇精品久久久久久| 91亚洲国产成人精品性色| 狠狠鲁狠狠操天天晚上干干| 欧美综合婷婷欧美综合| 亚洲无码一区在线影院| 五十路熟女人妻一区二| 免费一级特黄特色大片在线观看| 好男人视频在线免费观看网站| 久碰精品少妇中文字幕av| 国产夫妻视频在线观看免费| 久久机热/这里只有| 一区二区三区四区中文| 欧美一级色视频美日韩| 人人妻人人澡人人爽人人dvl| 日本一二三中文字幕| 91人妻人人做人人爽在线| 青娱乐最新视频在线| 香蕉aⅴ一区二区三区| 久久久久久国产精品| 成人av天堂丝袜在线观看| 欧美3p在线观看一区二区三区| 熟女人妻一区二区精品视频| 亚洲av日韩av第一区二区三区| 91精品资源免费观看| 亚洲一区久久免费视频| 成人高潮aa毛片免费| 亚洲天堂第一页中文字幕 | 亚洲图库另类图片区| 日韩一区二区电国产精品| 大香蕉日本伊人中文在线| 国产性感美女福利视频| 亚洲午夜在线视频福利| 美洲精品一二三产区区别| 99久久中文字幕一本人| 亚洲天堂av最新网址| 色婷婷六月亚洲综合香蕉| 亚洲精品国偷自产在线观看蜜桃| 天天日天天干天天爱| 99热99re在线播放| 91国语爽死我了不卡| 亚欧在线视频你懂的| 91试看福利一分钟| 亚洲av自拍天堂网| 成年人免费看在线视频| 亚洲欧美一区二区三区电影| 黑人变态深video特大巨大| 亚洲成av人无码不卡影片一| 日韩成人性色生活片| 100%美女蜜桃视频| 国产老熟女伦老熟妇ⅹ| 亚洲人妻av毛片在线| yy96视频在线观看| 国产精品成久久久久三级蜜臀av| 二区中出在线观看老师| 免费av岛国天堂网站| 免费看国产av网站| 在线观看一区二区三级| 中文字幕—97超碰网| 污污小视频91在线观看| 夜夜骑夜夜操夜夜奸| 亚洲午夜高清在线观看| 国产清纯美女al在线| 亚洲一区二区三区在线高清| 亚洲精品 欧美日韩| 黄色三级网站免费下载| 亚洲av在线观看尤物| 午夜频道成人在线91| 涩爱综合久久五月蜜臀| 操操网操操伊剧情片中文字幕网| 欧美日韩国产一区二区三区三州 | 天天射夜夜操综合网| 社区自拍揄拍尻屁你懂的| 1000部国产精品成人观看视频| 2022中文字幕在线| 丰满少妇翘臀后进式| 2018最新中文字幕在线观看| 涩爱综合久久五月蜜臀| 40道精品招牌菜特色| 亚洲久久午夜av一区二区| 少妇露脸深喉口爆吞精| 2022国产综合在线干| 日本少妇人妻xxxxxhd| 亚洲福利精品福利精品福利| 天天日天天添天天爽| 久久久久久久亚洲午夜综合福利| 韩国三级aaaaa高清视频| 欧美一级视频一区二区| 久久精品国产亚洲精品166m| 亚洲精品欧美日韩在线播放| 中文字幕在线观看极品视频| 日韩二区视频一线天婷婷五| 在线观看免费视频网| 国产成人精品av网站| 顶级尤物粉嫩小尤物网站| 国产精彩福利精品视频| 国产麻豆91在线视频| 中文字幕av一区在线观看| 国产视频网站一区二区三区 | 日韩欧美中文国产在线| 久久永久免费精品人妻专区| 国产麻豆剧传媒精品国产av蜜桃| 欧美美女人体视频一区| 亚洲伊人av天堂有码在线| av亚洲中文天堂字幕网| 男生舔女生逼逼的视频| 青青青国产片免费观看视频| 又粗又硬又猛又黄免费30| 大屁股熟女一区二区三区| 中文字幕+中文字幕| 成人网18免费视频版国产| 日本一道二三区视频久久| 97精品成人一区二区三区| 好吊视频—区二区三区| chinese国产盗摄一区二区| 爱有来生高清在线中文字幕| 亚洲av色图18p| 亚洲人妻视频在线网| 骚逼被大屌狂草视频免费看| 青青青视频手机在线观看| 亚洲熟妇无码一区二区三区| 偷青青国产精品青青在线观看| 在线观看亚洲人成免费网址| 伊人网中文字幕在线视频| 欧美日韩情色在线观看| 视频一区二区综合精品| 狠狠的往里顶撞h百合| 亚国产成人精品久久久| 色综合色综合色综合色| 中文字幕综合一区二区| 美女视频福利免费看| 40道精品招牌菜特色| 91色秘乱一区二区三区| 偷拍自拍亚洲美腿丝袜| 亚洲黄色av网站免费播放| 中文字幕—97超碰网| av在线免费观看亚洲天堂| 日美女屁股黄邑视频| 黑人变态深video特大巨大| 免费观看国产综合视频| 无套猛戳丰满少妇人妻| 中文字幕在线永久免费播放| 99视频精品全部15| 亚洲视频在线视频看视频在线| 绝顶痉挛大潮喷高潮无码 | 青青青爽视频在线播放| 日本在线不卡免费视频| 蜜桃专区一区二区在线观看| 中文字幕最新久久久| 狠狠的往里顶撞h百合| 国产av一区2区3区| 一级黄片久久久久久久久| 欧美精品欧美极品欧美视频| 中文字幕国产专区欧美激情| 中文字幕一区二区人妻电影冢本| 欧美成人精品欧美一级黄色| 国产变态另类在线观看| 日韩不卡中文在线视频网站| 婷婷久久一区二区字幕网址你懂得 | 中文字幕—97超碰网| 亚欧在线视频你懂的| 亚洲精品午夜aaa久久| 高清一区二区欧美系列| 国产日韩精品免费在线| 欧美地区一二三专区| 精品高潮呻吟久久av| 美日韩在线视频免费看| 岛国青草视频在线观看| 成人网18免费视频版国产| 亚洲一级美女啪啪啪| 亚洲 自拍 色综合图| 日韩精品中文字幕播放| 国产性生活中老年人视频网站| 国产一区av澳门在线观看| 热久久只有这里有精品| 激情国产小视频在线| 色偷偷伊人大杳蕉综合网 | 欧美综合婷婷欧美综合| 不戴胸罩引我诱的隔壁的人妻| 久久午夜夜伦痒痒想咳嗽P| 久久精品美女免费视频| 免费av岛国天堂网站| 91麻豆精品秘密入口在线观看| lutube在线成人免费看| 色狠狠av线不卡香蕉一区二区 | 亚洲天堂成人在线观看视频网站| 欧美一区二区三区久久久aaa| 九九热99视频在线观看97| 国产精品熟女久久久久浪潮| 日韩欧美一级黄片亚洲| www,久久久,com| 97超碰最新免费在线观看| 黄色大片男人操女人逼| 青草青永久在线视频18| 成年美女黄网站18禁久久| 天天日天天天天天天天天天天| 国产精彩福利精品视频| 欧美怡红院视频在线观看| 青青青青在线视频免费观看| 天天日天天舔天天射进去| 亚洲av琪琪男人的天堂| 亚洲国产成人av在线一区| 99热99re在线播放| 日本少妇的秘密免费视频| 国产高清在线在线视频| 精产国品久久一二三产区区别| 久青青草视频手机在线免费观看| 国产老熟女伦老熟妇ⅹ| 狠狠躁夜夜躁人人爽天天天天97| 老鸭窝日韩精品视频观看| 亚洲色偷偷综合亚洲AV伊人| 青青青青草手机在线视频免费看| 香蕉aⅴ一区二区三区| 五十路熟女人妻一区二| 日本免费视频午夜福利视频| 欧美3p在线观看一区二区三区| 大屁股肉感人妻中文字幕在线| 强行扒开双腿猛烈进入免费版| 啊啊啊想要被插进去视频| 五十路熟女av天堂| 国产麻豆剧果冻传媒app| 扒开让我视频在线观看| 天天色天天操天天舔| 亚洲av色香蕉一区二区三区| 日本一区美女福利视频| 沈阳熟妇28厘米大战黑人| 免费岛国喷水视频在线观看| 亚洲国产美女一区二区三区软件| 手机看片福利盒子日韩在线播放| 国产精品系列在线观看一区二区| 超碰97免费人妻麻豆| 自拍偷拍 国产资源| 国产免费高清视频视频| 亚洲成人国产av在线| 99精品视频之69精品视频 | 久久久久久久久久久久久97| 日韩熟女av天堂系列| 亚洲欧美色一区二区| 亚洲精品 日韩电影| 夏目彩春在线中文字幕| 999久久久久999| 亚洲伊人av天堂有码在线| 国产精品日韩欧美一区二区| 国产妇女自拍区在线观看| 国产污污污污网站在线| 操日韩美女视频在线免费看| 日本午夜爽爽爽爽爽视频在线观看| 亚洲av香蕉一区区二区三区犇| 国产卡一卡二卡三乱码手机| 噜噜色噜噜噜久色超碰| 精品高潮呻吟久久av| 欧美黄片精彩在线免费观看| 岛国av高清在线成人在线| 北条麻妃肉色丝袜视频| 91九色国产熟女一区二区| 欧美视频一区免费在线| 亚洲综合另类欧美久久| 人妻无码色噜噜狠狠狠狠色| 五十路人妻熟女av一区二区| 国产清纯美女al在线| 制服丝袜在线人妻中文字幕| 久久亚洲天堂中文对白| 97资源人妻免费在线视频| 亚洲成人黄色一区二区三区| 91福利在线视频免费观看| 成人综合亚洲欧美一区| 玖玖一区二区在线观看| 五月天色婷婷在线观看视频免费| 色偷偷伊人大杳蕉综合网| 自拍偷拍日韩欧美亚洲| 亚欧在线视频你懂的| 手机看片福利盒子日韩在线播放| 欧美成人综合色在线噜噜| 亚洲av无乱一区二区三区性色| 丝袜国产专区在线观看| 亚洲精品乱码久久久本| 久久一区二区三区人妻欧美| 夜夜骑夜夜操夜夜奸| 亚洲午夜电影之麻豆| 日韩在线中文字幕色| 亚洲综合乱码一区二区| 国产精品入口麻豆啊啊啊| 淫秽激情视频免费观看| 国产视频网站一区二区三区| 国产老熟女伦老熟妇ⅹ| 欧美在线精品一区二区三区视频| 97小视频人妻一区二区| 任你操任你干精品在线视频| 精品国产成人亚洲午夜| 欧美色婷婷综合在线| 日韩成人综艺在线播放| 天码人妻一区二区三区在线看| 亚洲av第国产精品| 天天色天天操天天舔| 青青青青青免费视频| 沙月文乃人妻侵犯中文字幕在线 | 亚洲精品国产久久久久久| 亚洲精品国产综合久久久久久久久| 青青草原网站在线观看| 五十路在线观看完整版| 午夜激情久久不卡一区二区 | 亚洲国产成人最新资源| av一本二本在线观看| 精品成人午夜免费看| 免费在线黄色观看网站| 色婷婷精品大在线观看| 漂亮 人妻被中出中文| 青青伊人一精品视频| 亚洲男人让女人爽的视频| 国产精品亚洲а∨天堂免| 青青在线视频性感少妇和隔壁黑丝 | 欧美韩国日本国产亚洲| 色秀欧美视频第一页| 天堂v男人视频在线观看| 喷水视频在线观看这里只有精品| 久久久超爽一二三av| av中文在线天堂精品| 国产成人精品福利短视频| 大鸡巴操娇小玲珑的女孩逼| 欧美日韩v中文在线| 精品高跟鞋丝袜一区二区| 成年人黄视频在线观看| 福利视频一区二区三区筱慧| 国产精品国产精品一区二区| 久久久久久久99精品| 天天日天天干天天插舔舔| 国产亚洲视频在线观看| 欧美一区二区三区高清不卡tv| 五色婷婷综合狠狠爱| 久久久久91精品推荐99| 天天日天天敢天天干| 天天操天天插天天色| 成人亚洲精品国产精品| 日本韩国免费一区二区三区视频| 国产女人叫床高潮大片视频| 大白屁股精品视频国产| 2022中文字幕在线| 欧美老妇精品另类不卡片| 自拍偷拍vs一区二区三区| 操的小逼流水的文章| 青青青爽视频在线播放| 2022中文字幕在线| 夜夜嗨av蜜臀av| 久久久制服丝袜中文字幕| 五月激情婷婷久久综合网| 在线观看免费视频色97| 精品人妻每日一部精品| 真实国产乱子伦一区二区| 国产高清97在线观看视频| 把腿张开让我插进去视频 | 国产麻豆91在线视频| 无码中文字幕波多野不卡| 91国产资源在线视频| 国产激情av网站在线观看| 久久三久久三久久三久久| 午夜在线一区二区免费| 国产精品三级三级三级| 亚洲国产第一页在线观看| 国产精品视频资源在线播放 | 天堂资源网av中文字幕| 亚洲欧美福利在线观看| 91破解版永久免费| 国产精品探花熟女在线观看| 三上悠亚和黑人665番号| 天美传媒mv视频在线观看| 99精品免费久久久久久久久a| 久久艹在线观看视频| 91破解版永久免费| 日韩成人免费电影二区| 国产无遮挡裸体免费直播视频| 91麻豆精品91久久久久同性| 日本一道二三区视频久久| 日韩影片一区二区三区不卡免费| 亚洲人人妻一区二区三区| 动漫av网站18禁| 亚洲av日韩精品久久久| 九色porny九色9l自拍视频| 久久久久久久亚洲午夜综合福利| 天天摸天天亲天天舔天天操天天爽| 国产成人综合一区2区| 国产精品久久久久久久精品视频| 亚洲成人激情av在线| 欧美爆乳肉感大码在线观看| 91久久精品色伊人6882| 亚洲视频在线视频看视频在线| av一区二区三区人妻| 亚洲一区二区三区五区| 青青擦在线视频国产在线| 18禁精品网站久久| 午夜美女少妇福利视频| 亚洲码av无色中文| 毛茸茸的大外阴中国视频| 成年人的在线免费视频| 亚洲国产香蕉视频在线播放| 精内国产乱码久久久久久| 夜鲁夜鲁狠鲁天天在线| 91人妻精品一区二区在线看| 天天操天天干天天日狠狠插| 2021天天色天天干| 欧美视频中文一区二区三区| 骚货自慰被发现爆操| 亚洲国产在人线放午夜| 白嫩白嫩美女极品国产在线观看| 521精品视频在线观看| 午夜91一区二区三区| 福利视频网久久91| 国产大鸡巴大鸡巴操小骚逼小骚逼| 欧美爆乳肉感大码在线观看| 青青草亚洲国产精品视频| 在线观看视频 你懂的| 91在线视频在线精品3| 在线观看一区二区三级| 岳太深了紧紧的中文字幕| 久久久精品精品视频视频| 成人精品视频99第一页| 黑人借宿ntr人妻的沦陷2| 欧美一区二区三区久久久aaa| 欧洲日韩亚洲一区二区三区| 欧美精品 日韩国产| 日韩精品一区二区三区在线播放| 2021久久免费视频| 欧美乱妇无乱码一区二区| 亚洲国产香蕉视频在线播放| 久草视频中文字幕在线观看| 人人在线视频一区二区| 天天操天天干天天插| 成人精品在线观看视频| 亚洲av自拍偷拍综合| 大香蕉伊人中文字幕| 1024久久国产精品| www天堂在线久久| 久久美欧人妻少妇一区二区三区| 天天通天天透天天插| 亚洲国产精品免费在线观看| 中文字幕1卡1区2区3区| 亚洲欧美国产综合777| 五十路老熟女码av| 自拍偷拍亚洲精品第2页| 午夜美女少妇福利视频| 亚洲男人的天堂a在线| 91欧美在线免费观看| 亚洲一区二区三区精品乱码| 九色porny九色9l自拍视频| 色哟哟在线网站入口| 经典国语激情内射视频| 色哟哟国产精品入口| 亚洲av在线观看尤物| 天天操天天操天天碰| 小穴多水久久精品免费看| 五月天中文字幕内射| 国产性感美女福利视频| 国产女人被做到高潮免费视频 | 动漫av网站18禁| 亚洲精品乱码久久久久久密桃明| 国产之丝袜脚在线一区二区三区| 国产日韩一区二区在线看| av乱码一区二区三区| 欧美另类重口味极品在线观看| 欧美精品激情在线最新观看视频| 欧美精品欧美极品欧美视频| 韩国男女黄色在线观看| 女蜜桃臀紧身瑜伽裤| 免费十精品十国产网站| 欧美另类一区二区视频| 国产1区,2区,3区| 看一级特黄a大片日本片黑人| 日日爽天天干夜夜操| 精品美女久久久久久| 大鸡巴操b视频在线| 大香蕉日本伊人中文在线| 水蜜桃国产一区二区三区| 男人的天堂在线黄色| 91试看福利一分钟| 伊人成人综合开心网| 男大肉棒猛烈插女免费视频| 我想看操逼黄色大片| 亚洲少妇高潮免费观看| 亚洲 中文 自拍 无码| 久久尻中国美女视频| 欧美日本aⅴ免费视频| 中文字幕综合一区二区| 孕妇奶水仑乱A级毛片免费看| 91传媒一区二区三区| 精品老妇女久久9g国产| 抽查舔水白紧大视频| 97资源人妻免费在线视频| 国产黑丝高跟鞋视频在线播放| 在线视频精品你懂的| 日本一区精品视频在线观看| 久久国产精品精品美女| 中文字幕AV在线免费看 | 福利视频一区二区三区筱慧| 精品久久久久久高潮| 99久久久无码国产精品性出奶水| 亚洲偷自拍高清视频| 日本在线不卡免费视频| 日韩一个色综合导航| 国产九色91在线视频| 97超碰最新免费在线观看| v888av在线观看视频| 91小伙伴中女熟女高潮| 亚洲成高清a人片在线观看| 美女福利视频网址导航| 巨乳人妻日下部加奈被邻居中出| 国产女人露脸高潮对白视频| 精品日产卡一卡二卡国色天香| 亚洲精品高清自拍av| 91国语爽死我了不卡| 亚洲天堂精品久久久| 狠狠躁夜夜躁人人爽天天天天97| 自拍偷拍,中文字幕| 日韩av中文在线免费观看| av网站色偷偷婷婷网男人的天堂| 99精品久久久久久久91蜜桃| 久久这里只有精品热视频 | 国产真实乱子伦a视频| 亚洲高清免费在线观看视频| 亚洲在线一区二区欧美| av资源中文字幕在线观看| 玩弄人妻熟妇性色av少妇| 亚洲 中文 自拍 无码| 女人精品内射国产99| 激情图片日韩欧美人妻| 91小伙伴中女熟女高潮| 日本丰满熟妇BBXBBXHD| 亚洲 图片 欧美 图片| 国产精品国产精品一区二区| 日本熟女50视频免费| av欧美网站在线观看| 久久国产精品精品美女| 亚洲va欧美va人人爽3p| jiujiure精品视频在线| 国产精品黄大片在线播放| 在线免费观看视频一二区| 999九九久久久精品| av破解版在线观看| 欧美成人黄片一区二区三区| 男人靠女人的逼视频| 亚洲中文字幕国产日韩| 适合午夜一个人看的视频| 成人国产小视频在线观看| 19一区二区三区在线播放| 亚洲日本一区二区三区| 色天天天天射天天舔| 精品久久久久久久久久久a√国产| av中文字幕福利网| av新中文天堂在线网址| 人人超碰国字幕观看97| 国产黄网站在线观看播放| 日本熟女50视频免费| 2020国产在线不卡视频| 欧美视频中文一区二区三区| 中文字幕无码一区二区免费| 欧美韩国日本国产亚洲| 国产在线观看免费人成短视频| 精品区一区二区三区四区人妻| 男女第一次视频在线观看| 中文字幕中文字幕 亚洲国产| 51精品视频免费在线观看| 在线视频自拍第三页| 唐人色亚洲av嫩草| 亚洲1069综合男同| 大胆亚洲av日韩av| 中文字母永久播放1区2区3区| 男人的天堂av日韩亚洲| 国产精选一区在线播放| 青青青青视频在线播放| 蜜桃视频在线欧美一区| 午夜在线精品偷拍一区二| 亚洲精品在线资源站| 国产欧美精品一区二区高清| 一区二区三区精品日本| 免费看高清av的网站| 99精品久久久久久久91蜜桃| 在线视频精品你懂的| 欧美成人精品欧美一级黄色| 操的小逼流水的文章| 男女之间激情网午夜在线| 91片黄在线观看喷潮| 最新91九色国产在线观看| 在线播放一区二区三区Av无码| 成人区人妻精品一区二视频| 后入美女人妻高清在线| 亚洲高清免费在线观看视频| 97超碰最新免费在线观看| 亚洲欧美精品综合图片小说| 中文字幕之无码色多多| 欧美中文字幕一区最新网址| 亚洲中文字幕综合小综合| 91成人精品亚洲国产| 日韩欧美国产一区不卡| 超碰公开大香蕉97| 四川乱子伦视频国产vip| 白白操白白色在线免费视频| 北条麻妃av在线免费观看| 少妇深喉口爆吞精韩国| 亚洲av第国产精品| 一二三区在线观看视频| 天天摸天天日天天操| 中文字幕在线乱码一区二区| 少妇人妻真实精品视频| 大陆精品一区二区三区久久| 亚洲精品午夜aaa久久| 337p日本大胆欧美人| 天堂av在线官网中文| 亚洲激情唯美亚洲激情图片| 亚洲av无女神免非久久| 中文字幕av一区在线观看| 一区二区视频视频视频| 色综合天天综合网国产成人| av新中文天堂在线网址| 亚洲国际青青操综合网站| 日本少妇人妻xxxxxhd| 国产亚洲国产av网站在线| 亚洲精品av在线观看| 噜噜色噜噜噜久色超碰| 亚洲另类综合一区小说| 日本免费一级黄色录像| 欧美视频不卡一区四区| 国产chinesehd精品麻豆| 日韩欧美国产精品91| 日韩欧美中文国产在线| 亚洲成人三级在线播放| 精品区一区二区三区四区人妻 | 91大屁股国产一区二区| 自拍偷拍亚洲另类色图| 亚洲另类在线免费观看| 视频二区在线视频观看| 欧美一区二区中文字幕电影 | 国产黄色片在线收看| 国产97视频在线精品| 亚洲综合一区二区精品久久| 免费成人va在线观看| 日韩欧美一级黄片亚洲| 丝袜长腿第一页在线| 丰满的子国产在线观看| 天天干天天日天天干天天操| 亚洲免费福利一区二区三区| 亚洲欧美综合在线探花| 91试看福利一分钟| 超碰97人人澡人人| 午夜精品久久久久麻豆影视| 精内国产乱码久久久久久| 97人妻人人澡爽人人精品| 天天日天天透天天操| 制服丝袜在线人妻中文字幕| 久久这里有免费精品| 日本一二三中文字幕| 国产视频在线视频播放| 一个人免费在线观看ww视频| 91国偷自产一区二区三区精品| 69精品视频一区二区在线观看| 青青青青草手机在线视频免费看| 91欧美在线免费观看| 视频在线亚洲一区二区| 国产福利小视频大全| 久久精品视频一区二区三区四区 | 久久麻豆亚洲精品av| 天天摸天天亲天天舔天天操天天爽| av手机免费在线观看高潮| 999热精品视频在线| 极品丝袜一区二区三区| 中文字幕免费福利视频6| 免费看国产av网站| 免费十精品十国产网站| 中国把吊插入阴蒂的视频| 人妻丝袜诱惑我操她视频| 国产剧情演绎系列丝袜高跟| 91老师蜜桃臀大屁股| 大香蕉日本伊人中文在线| 天天日夜夜干天天操| 国产91精品拍在线观看| 午夜在线观看岛国av,com| 91在线免费观看成人| 硬鸡巴动态操女人逼视频| 国产乱子伦精品视频潮优女| 91色老99久久九九爱精品| 91人妻精品一区二区久久| 91天堂精品一区二区| 久久久久久久一区二区三| 亚洲视频在线观看高清| 久久免看30视频口爆视频| 男人和女人激情视频| 亚洲av黄色在线网站| 五十路在线观看完整版| 国内资源最丰富的网站| 一个色综合男人天堂| 91国内精品久久久久精品一| 黄色无码鸡吧操逼视频| 亚洲精品国产久久久久久| 精品高跟鞋丝袜一区二区| 精品亚洲在线免费观看| 日本女人一级免费片| 婷婷色国产黑丝少妇勾搭AV| 欧美另类重口味极品在线观看| 亚洲成人免费看电影| 成年人啪啪视频在线观看| 桃色视频在线观看一区二区 | 欧美另类一区二区视频| 青青伊人一精品视频| 亚洲免费视频欧洲免费视频| 边摸边做超爽毛片18禁色戒| 欧美一区二区三区乱码在线播放| 国产九色91在线视频| av高潮迭起在线观看| 日韩精品中文字幕播放| 亚洲精品无码色午夜福利理论片| av日韩在线观看大全| 国产精品3p和黑人大战| 91精品综合久久久久3d动漫| 91欧美在线免费观看| 亚洲av可乐操首页| 和邻居少妇愉情中文字幕| 亚洲综合乱码一区二区| 91国产资源在线视频| 国产日韩精品电影7777| 激情啪啪啪啪一区二区三区| 色婷婷综合激情五月免费观看| 日本18禁久久久久久| 亚洲av无女神免非久久| 国产精品人妻66p| 77久久久久国产精产品| 鸡巴操逼一级黄色气| 国产日韩欧美视频在线导航| 国产成人自拍视频播放| 美女小视频网站在线| yy96视频在线观看| 中国黄色av一级片| 热99re69精品8在线播放| 亚洲成人免费看电影| 好了av中文字幕在线| 国产夫妻视频在线观看免费| 天天插天天色天天日| 又色又爽又黄的美女裸体| 免费观看国产综合视频| 久久尻中国美女视频| 亚洲av日韩av第一区二区三区| 亚洲专区激情在线观看视频| 亚洲综合在线观看免费| 久久三久久三久久三久久| 福利视频网久久91| 好太好爽好想要免费| 国产在线观看黄色视频| 自拍偷拍亚洲精品第2页| 黄网十四区丁香社区激情五月天| av亚洲中文天堂字幕网| 国产露脸对白在线观看| 狠狠躁夜夜躁人人爽天天久天啪| 日韩精品啪啪视频一道免费| 一色桃子久久精品亚洲| 五十路熟女人妻一区二| 亚洲护士一区二区三区| 91九色porny国产在线| 粉嫩av懂色av蜜臀av| 日韩午夜福利精品试看| 国产V亚洲V天堂无码欠欠| 国产成人一区二区三区电影网站| 亚洲国产第一页在线观看| 亚洲美女自偷自拍11页| 青青青青青青草国产| 亚洲1069综合男同| 欧美一区二区三区啪啪同性| 中国黄片视频一区91| 国产自拍在线观看成人| 中国黄片视频一区91| av破解版在线观看| 狠狠操操操操操操操操操| 精品国产成人亚洲午夜| 黑人乱偷人妻中文字幕| 在线免费91激情四射| 蜜臀av久久久久蜜臀av麻豆| 欧美日韩熟女一区二区三区| 桃色视频在线观看一区二区| 国产一区二区欧美三区| 亚洲精品亚洲人成在线导航| 天天日天天做天天日天天做| 中文字幕无码一区二区免费| 自拍偷拍亚洲精品第2页| 9国产精品久久久久老师| 青草青永久在线视频18| 免费大片在线观看视频网站| 午夜久久久久久久99| 亚洲欧美清纯唯美另类| 91国内精品自线在拍白富美| 东京热男人的av天堂| 国产精品自偷自拍啪啪啪| 青青青青青手机视频| 中文字幕一区二 区二三区四区| 亚洲欧美综合在线探花| 欧美中文字幕一区最新网址| 日韩在线视频观看有码在线| 国产综合高清在线观看| 大鸡巴后入爆操大屁股美女| av中文字幕在线导航| 91av精品视频在线| 人人妻人人澡人人爽人人dvl| 久久亚洲天堂中文对白| 偷拍自拍亚洲视频在线观看| 自拍偷拍日韩欧美一区二区| 91国语爽死我了不卡| 亚洲欧美清纯唯美另类| 亚洲精品午夜aaa久久| 日本精品视频不卡一二三| 999九九久久久精品| 中文字幕一区二区自拍| 欧美亚洲少妇福利视频| av视屏免费在线播放| 十八禁在线观看地址免费| 日比视频老公慢点好舒服啊| 日本欧美视频在线观看三区| 亚洲欧洲av天堂综合| 99国产精品窥熟女精品| japanese日本熟妇另类| 极品粉嫩小泬白浆20p主播| 高清成人av一区三区| 岛国av高清在线成人在线| 日本一本午夜在线播放| 国产成人无码精品久久久电影| 在线免费观看视频一二区| 日韩精品二区一区久久| 爱有来生高清在线中文字幕| 白嫩白嫩美女极品国产在线观看| 涩涩的视频在线观看视频| 香蕉aⅴ一区二区三区| 欧美一区二区三区久久久aaa| 免费国产性生活视频| 天天日天天日天天射天天干| 大白屁股精品视频国产| 四虎永久在线精品免费区二区| 亚洲精品一区二区三区老狼| 最新91九色国产在线观看| 中国无遮挡白丝袜二区精品| 在线国产日韩欧美视频| 视频一区二区三区高清在线| 国产在线拍揄自揄视频网站| 中文字幕在线第一页成人| 久草视频中文字幕在线观看| 欧美成人精品在线观看| 2022国产精品视频| 男人操女人的逼免费视频| 2020久久躁狠狠躁夜夜躁| 91精品国产黑色丝袜| 亚洲欧洲av天堂综合| 99热国产精品666| 老鸭窝日韩精品视频观看| 国产乱子伦一二三区| 北条麻妃av在线免费观看| 天天色天天操天天舔| 日韩av中文在线免费观看| 黑人解禁人妻叶爱071| 亚洲欧美成人综合视频| 性感美女诱惑福利视频| 天天日天天干天天要| 天天插天天色天天日| 77久久久久国产精产品| 亚洲人妻av毛片在线| 青春草视频在线免费播放| 人妻久久无码中文成人| 精产国品久久一二三产区区别| 午夜精品九一唐人麻豆嫩草成人| 91国内精品自线在拍白富美| 亚洲国产成人最新资源| 3344免费偷拍视频| 亚洲的电影一区二区三区| 国产亚洲成人免费在线观看| 在线免费观看av日韩| 大胸性感美女羞爽操逼毛片| 绯色av蜜臀vs少妇| 国产精品久久久久久美女校花| 国产aⅴ一线在线观看| 亚洲中文字幕校园春色 | 人妻丝袜诱惑我操她视频| 亚洲精品在线资源站| 亚洲天天干 夜夜操| 久久久久久久精品成人热| 视频一区二区在线免费播放| 日本一区精品视频在线观看| 国产午夜激情福利小视频在线| 免费国产性生活视频| 75国产综合在线视频| 日本高清成人一区二区三区| 国产91嫩草久久成人在线视频| 成人30分钟免费视频| 最新97国产在线视频| 欧美爆乳肉感大码在线观看| 日韩中文字幕精品淫| 日本18禁久久久久久| 亚洲中文字幕综合小综合| 国产V亚洲V天堂无码欠欠| 日韩国产乱码中文字幕| 午夜国产福利在线观看| 精品国产高潮中文字幕| 日本男女操逼视频免费看 | 亚洲av色图18p| 国产精品一区二区av国| 亚洲中文字幕乱码区| 青青青青青青青青青国产精品视频| 大鸡八强奸视频在线观看| 青青青青爽手机在线| 婷婷久久久久深爱网| 国产精品一区二区av国| 精品一线二线三线日本| 成人福利视频免费在线| 91精品国产黑色丝袜| 亚洲国产欧美一区二区丝袜黑人| 视频一区二区在线免费播放 | 人妻少妇一区二区三区蜜桃| 天天操夜夜骑日日摸| 夜夜操,天天操,狠狠操| 欧美另类重口味极品在线观看| 色秀欧美视频第一页| 在线观看操大逼视频| 午夜福利资源综合激情午夜福利资| 色秀欧美视频第一页| 国产一区二区三免费视频 | 99精品视频之69精品视频| 三级等保密码要求条款| 日本在线一区二区不卡视频| 日韩欧美在线观看不卡一区二区| 熟女少妇激情五十路| 把腿张开让我插进去视频| 含骚鸡巴玩逼逼视频| 日韩熟女av天堂系列| 沙月文乃人妻侵犯中文字幕在线| 天天摸天天亲天天舔天天操天天爽| 青娱乐最新视频在线| 社区自拍揄拍尻屁你懂的| 天天干狠狠干天天操| 美女小视频网站在线| 精品人妻一二三区久久| 亚洲最大黄了色网站| 丝袜肉丝一区二区三区四区在线看| 亚洲欧美福利在线观看| 亚洲av自拍天堂网| 五十路熟女av天堂| 人妻另类专区欧美制服| 亚洲va国产va欧美精品88| 国产成人午夜精品福利| 日韩伦理短片在线观看| 五月天中文字幕内射| 一个人免费在线观看ww视频| 青青青青青青青在线播放视频| 偷青青国产精品青青在线观看| 亚洲丝袜老师诱惑在线观看| 国产久久久精品毛片| 国产精品久久久黄网站| 伊人开心婷婷国产av| 精品首页在线观看视频| 中文字幕第1页av一天堂网| 国产刺激激情美女网站| 午夜福利资源综合激情午夜福利资 | 视频在线亚洲一区二区| 亚洲美女美妇久久字幕组| 欧美老鸡巴日小嫩逼| 抽查舔水白紧大视频| 午夜精品福利91av| 国产精品一区二区av国| 巨乳人妻日下部加奈被邻居中出| 中文字幕一区二区三区蜜月| 亚洲av成人免费网站| 99的爱精品免费视频| 欧美伊人久久大香线蕉综合| 午夜青青草原网在线观看| 欧美亚洲一二三区蜜臀| 懂色av之国产精品| 亚洲国产免费av一区二区三区| 99久久激情婷婷综合五月天| 亚洲激情av一区二区| 中文字幕乱码人妻电影| 五月天中文字幕内射| www日韩毛片av| 在线免费观看日本伦理| 一色桃子久久精品亚洲| 少妇人妻二三区视频| 边摸边做超爽毛片18禁色戒| 日本xx片在线观看| 小泽玛利亚视频在线观看| 天天干夜夜操啊啊啊| 2020韩国午夜女主播在线| 伊人成人在线综合网| 中文 成人 在线 视频| 天天插天天狠天天操| av天堂中文字幕最新| 亚洲欧美人精品高清| 久草福利电影在线观看| 成人资源在线观看免费官网| av天堂中文字幕最新| 亚洲图片偷拍自拍区| 午夜成午夜成年片在线观看| 亚洲精品成人网久久久久久小说| 国产午夜激情福利小视频在线| 91一区精品在线观看| 午夜av一区二区三区| av老司机精品在线观看| 日本熟妇色熟妇在线观看| japanese五十路熟女熟妇| 在线免费观看欧美小视频| 一色桃子人妻一区二区三区| 国产变态另类在线观看| 丁香花免费在线观看中文字幕| 午夜精品久久久久久99热| 国产成人自拍视频播放| 经典国语激情内射视频| 宅男噜噜噜666免费观看| 看一级特黄a大片日本片黑人| 99国内精品永久免费视频| 91在线视频在线精品3| 白嫩白嫩美女极品国产在线观看| 国产欧美日韩第三页| 亚洲福利午夜久久久精品电影网| 久久久久91精品推荐99| 初美沙希中文字幕在线 | 成年人免费看在线视频| 亚洲一级美女啪啪啪| 瑟瑟视频在线观看免费视频| 国产日韩精品一二三区久久久| 国产亚洲成人免费在线观看| av在线免费观看亚洲天堂| 91欧美在线免费观看| 日本韩国在线观看一区二区| 亚洲 中文 自拍 无码| 蜜桃臀av蜜桃臀av| 80电影天堂网官网| 日本一区精品视频在线观看| 国产视频在线视频播放| 91天堂精品一区二区| 亚洲av第国产精品| 亚洲国产精品美女在线观看| 天天摸天天日天天操| 综合精品久久久久97| 97人人妻人人澡人人爽人人精品| 人人妻人人爽人人澡人人精品| 熟女俱乐部一二三区| 黑人变态深video特大巨大| 久久丁香花五月天色婷婷| 懂色av之国产精品| 男生舔女生逼逼视频| 久草极品美女视频在线观看| 国产97在线视频观看| 久草视频在线一区二区三区资源站| 亚洲精品无码久久久久不卡| 高清成人av一区三区| 成人资源在线观看免费官网| avjpm亚洲伊人久久| 一色桃子久久精品亚洲| 91中文字幕免费在线观看| 极品粉嫩小泬白浆20p主播| 黄工厂精品视频在线观看| 亚洲成人激情视频免费观看了| 日韩不卡中文在线视频网站| 国产av国片精品一区二区| 青青青青视频在线播放| 操的小逼流水的文章| 国产麻豆剧传媒精品国产av蜜桃| 人妻少妇精品久久久久久| 欲乱人妻少妇在线视频裸| 五十路在线观看完整版| 亚洲va天堂va国产va久| 欧美在线偷拍视频免费看| 又黄又刺激的午夜小视频| 中字幕人妻熟女人妻a62v网| 色狠狠av线不卡香蕉一区二区| 我想看操逼黄色大片| 自拍偷拍日韩欧美亚洲| 无码日韩人妻精品久久| 欧美亚洲中文字幕一区二区三区| 人妻av无码专区久久绿巨人| 欧美成人综合视频一区二区| 91国语爽死我了不卡| 91福利在线视频免费观看| 91精品高清一区二区三区| 亚洲高清国产拍青青草原| 国产精品一区二区av国| 99久久久无码国产精品性出奶水 | 亚洲av午夜免费观看| 亚洲精品 欧美日韩| 日韩欧美中文国产在线| 欧美精品国产综合久久| 黄色视频在线观看高清无码| 丰满少妇人妻xxxxx| 91一区精品在线观看| 亚洲一区二区三区偷拍女厕91 | 日韩欧美高清免费在线| 日韩av有码中文字幕| 成人综合亚洲欧美一区| 日本性感美女视频网站| 国产成人自拍视频在线免费观看| 超碰在线观看免费在线观看| av中文字幕国产在线观看| 99精品国产aⅴ在线观看| 黑人大几巴狂插日本少妇| 日韩成人性色生活片| 97精品成人一区二区三区| 成人影片高清在线观看| 2022国产综合在线干| 黄页网视频在线免费观看| 色哟哟在线网站入口| 日本熟妇色熟妇在线观看| 国产视频网站一区二区三区| 日韩人妻在线视频免费| 国产亚洲天堂天天一区| av中文字幕电影在线看| 免费一级特黄特色大片在线观看| 亚洲熟女久久久36d| 黄色大片男人操女人逼| 欧美精品激情在线最新观看视频| 欧美一区二区三区四区性视频| 亚洲天堂av最新网址| av在线免费观看亚洲天堂| 888欧美视频在线| 最新91精品视频在线| 成年人啪啪视频在线观看| 日本熟女精品一区二区三区| 97a片免费在线观看| 欧美女同性恋免费a| 啪啪啪啪啪啪啪啪啪啪黄色| 精品高潮呻吟久久av| 久久永久免费精品人妻专区 | 成人sm视频在线观看| 亚洲午夜福利中文乱码字幕| 久久久久久国产精品| 五月天色婷婷在线观看视频免费| 国产精品国产三级麻豆| 超级福利视频在线观看| 亚洲熟妇久久无码精品| 五十路丰满人妻熟妇| 黄网十四区丁香社区激情五月天 | 久久久久久9999久久久久| 揄拍成人国产精品免费看视频| 国产精品探花熟女在线观看| 熟女人妻三十路四十路人妻斩| 天天日天天透天天操| 精品国产污污免费网站入口自| 欧美一区二区三区在线资源 | 国产女人被做到高潮免费视频 | 亚洲综合图片20p| 亚洲成人av一区久久| 中文字幕奴隷色的舞台50| 亚洲高清国产一区二区三区| 亚洲国产40页第21页| 人妻丝袜榨强中文字幕| 国产成人综合一区2区| 动漫黑丝美女的鸡巴| 亚洲粉嫩av一区二区三区| 国产女人被做到高潮免费视频| 欧美熟妇一区二区三区仙踪林| 天天干天天插天天谢| 91亚洲国产成人精品性色| 性感美女高潮视频久久久| 福利视频一区二区三区筱慧| 小穴多水久久精品免费看| 天天艹天天干天天操| 婷婷久久一区二区字幕网址你懂得| 大白屁股精品视频国产| 婷婷六月天中文字幕| 日本免费午夜视频网站| 中文字幕 码 在线视频| 天天综合天天综合天天网| 晚上一个人看操B片| av乱码一区二区三区| 日本丰满熟妇大屁股久久| 香港三日本三韩国三欧美三级| 好男人视频在线免费观看网站| 欧美精品黑人性xxxx| 五十路熟女人妻一区二| 亚洲另类伦春色综合小| 夫妻在线观看视频91| 欧美专区第八页一区在线播放| 最后99天全集在线观看| 日日操夜夜撸天天干| 高清成人av一区三区| av高潮迭起在线观看| 91国产在线视频免费观看| 亚洲欧美清纯唯美另类| 国产又粗又硬又大视频| 国产高清精品极品美女| 中国视频一区二区三区| 色哟哟国产精品入口| 欧美性受xx黑人性猛交| 国产又色又刺激在线视频| 精品黑人巨大在线一区| 伊人网中文字幕在线视频| 精品人妻一二三区久久| 在线观看免费视频色97| 97人妻色免费视频| 免费费一级特黄真人片 | 在线亚洲天堂色播av电影| 欧美精产国品一二三区| 福利一二三在线视频观看| 国产成人一区二区三区电影网站 | 91九色porny国产在线| 人人爽亚洲av人人爽av| 2020国产在线不卡视频| 成人亚洲精品国产精品| 国产aⅴ一线在线观看| 成年人午夜黄片视频资源| 护士特殊服务久久久久久久| 人妻少妇av在线观看| 肏插流水妹子在线乐播下载| 日本xx片在线观看| 亚洲午夜精品小视频| 国产视频一区在线观看| 视频在线免费观看你懂得| 亚洲av日韩高清hd| 青青青青操在线观看免费| 天天日天天操天天摸天天舔 | 性感美女诱惑福利视频| 欧美精品伦理三区四区| 狠狠躁狠狠爱网站视频| 97a片免费在线观看| 熟女国产一区亚洲中文字幕| 日韩精品啪啪视频一道免费| 日日操夜夜撸天天干| 四虎永久在线精品免费区二区| 国产亚州色婷婷久久99精品| 中国老熟女偷拍第一页| 欧美精产国品一二三产品区别大吗| 亚洲最大免费在线观看| 爱有来生高清在线中文字幕| 国产精品大陆在线2019不卡| 特一级特级黄色网片| 成年人中文字幕在线观看| 黄色录像鸡巴插进去| 亚洲欧美日韩视频免费观看| 国产午夜激情福利小视频在线| 精品久久久久久久久久中文蒉| 欧美亚洲中文字幕一区二区三区| 亚洲av天堂在线播放| 98精产国品一二三产区区别| 中文字幕视频一区二区在线观看| 人妻在线精品录音叫床| 不卡一不卡二不卡三| 精品一区二区三区三区88| 视频二区在线视频观看| 天天日天天玩天天摸| 天天日天天干天天干天天日| 国产精品系列在线观看一区二区| 国产自拍在线观看成人| 免费福利av在线一区二区三区| 亚洲Av无码国产综合色区| 亚洲老熟妇日本老妇| 超级av免费观看一区二区三区| 午夜久久久久久久精品熟女| 99av国产精品欲麻豆| 2020国产在线不卡视频| 亚洲精品午夜久久久久| 狠狠的往里顶撞h百合| 国产福利小视频大全| 蜜桃色婷婷久久久福利在线| 欧美日韩一区二区电影在线观看| 男女第一次视频在线观看| 夜鲁夜鲁狠鲁天天在线| 国产变态另类在线观看| 性欧美日本大妈母与子| 4个黑人操素人视频网站精品91| 成人网18免费视频版国产| 夜夜骑夜夜操夜夜奸| 欧美另类重口味极品在线观看| 欧美日韩精品永久免费网址| 99精品免费久久久久久久久a| 人人妻人人爱人人草| 国产精品久久9999| 欧美日韩在线精品一区二区三| 日本熟女精品一区二区三区| 天天躁日日躁狠狠躁躁欧美av| 欧美日韩中文字幕欧美| 在线免费观看靠比视频的网站| 99热99这里精品6国产| 国产熟妇乱妇熟色T区| 久久这里只有精品热视频| 亚洲av无码成人精品区辽| av在线免费观看亚洲天堂| 国产第一美女一区二区三区四区 | 在线观看的a站 最新| 亚洲日本一区二区久久久精品| 9色精品视频在线观看| 阴茎插到阴道里面的视频| 91九色porny国产在线| 国产又粗又猛又爽又黄的视频在线| 天堂女人av一区二区| 免费黄色成人午夜在线网站| 黄色男人的天堂视频| 丰满少妇人妻xxxxx| 国产清纯美女al在线| 国产白袜脚足J棉袜在线观看| 久久精品国产亚洲精品166m| 国产av自拍偷拍盛宴| 91在线视频在线精品3| 一区二区熟女人妻视频| 日本熟妇一区二区x x| 淫秽激情视频免费观看| 国产麻豆精品人妻av| 精品91高清在线观看| 99精品一区二区三区的区| 成年人黄色片免费网站| 天天日天天摸天天爱| av久久精品北条麻妃av观看| 日韩精品二区一区久久| 在线视频免费观看网| 日韩欧美一级黄片亚洲| 成年美女黄网站18禁久久| 毛片一级完整版免费| 粉嫩av蜜乳av蜜臀| 欧美麻豆av在线播放| 免费在线看的黄网站| caoporm超碰国产| 亚洲1卡2卡三卡4卡在线观看 | 亚洲熟妇无码一区二区三区| 把腿张开让我插进去视频| 97人妻色免费视频| 中文字幕在线永久免费播放| 欧洲欧美日韩国产在线| 免费大片在线观看视频网站| 国产又粗又硬又大视频| 晚上一个人看操B片| 亚洲va欧美va人人爽3p| 亚洲男人在线天堂网| 亚洲特黄aaaa片| 69精品视频一区二区在线观看| 中文字幕第三十八页久久| 国产三级精品三级在线不卡| 国产福利小视频二区| 欧美国产亚洲中英文字幕| 欧美日韩情色在线观看| 国产亚州色婷婷久久99精品| 2021最新热播中文字幕| 9l人妻人人爽人人爽| 欧美在线偷拍视频免费看| 偷拍自拍亚洲美腿丝袜| 91九色porny国产在线| 久久久久久9999久久久久| 日韩黄色片在线观看网站| 青青在线视频性感少妇和隔壁黑丝| 国产av福利网址大全| 性感美女诱惑福利视频| 色噜噜噜噜18禁止观看| 亚洲av在线观看尤物| 久久精品在线观看一区二区| 夜色17s精品人妻熟女| 国产露脸对白在线观看| 沈阳熟妇28厘米大战黑人| 日本成人一区二区不卡免费在线| 欧美国产亚洲中英文字幕| 亚洲国产最大av综合| 天天日天天爽天天干| 夏目彩春在线中文字幕| 自拍偷拍vs一区二区三区| 亚洲另类图片蜜臀av| 黄片三级三级三级在线观看| 福利在线视频网址导航| 久久机热/这里只有| 日韩精品中文字幕福利| 亚洲欧美一区二区三区电影| 男人操女人的逼免费视频| 女蜜桃臀紧身瑜伽裤| 91久久综合男人天堂| 最新日韩av传媒在线| av破解版在线观看| 午夜婷婷在线观看视频| asmr福利视频在线观看| 国产视频一区在线观看| av在线观看网址av| 亚洲av香蕉一区区二区三区犇| 美女骚逼日出水来了| 欧美老妇精品另类不卡片| 巨乳人妻日下部加奈被邻居中出| 中文字幕日韩精品就在这里| 国产精品中文av在线播放| 日本免费午夜视频网站| 91试看福利一分钟| 国产综合高清在线观看| 亚洲1卡2卡三卡4卡在线观看 | 亚洲午夜高清在线观看| 日本性感美女写真视频| 中文字幕在线观看极品视频| 国产精品成久久久久三级蜜臀av| 91九色porny蝌蚪国产成人| 天堂av狠狠操蜜桃| 最新97国产在线视频| 不戴胸罩引我诱的隔壁的人妻| 亚洲最大黄了色网站| 成人av天堂丝袜在线观看| 少妇高潮无套内谢麻豆| 亚洲av日韩高清hd| 亚洲精品 欧美日韩| 好吊操视频这里只有精品| 亚洲国产在人线放午夜| 男女第一次视频在线观看| 久久久久久性虐视频| 久久久精品欧洲亚洲av| 色婷婷久久久久swag精品| 93视频一区二区三区| 红桃av成人在线观看| 一区二区三区四区五区性感视频| 韩国三级aaaaa高清视频| 午夜dv内射一区区| 色噜噜噜噜18禁止观看| 亚洲一区二区三区偷拍女厕91| 99国内精品永久免费视频| 欧美视频不卡一区四区| 亚洲天堂有码中文字幕视频| 老鸭窝在线观看一区| 欧美亚洲中文字幕一区二区三区| 水蜜桃一区二区三区在线观看视频| 开心 色 六月 婷婷| 熟女俱乐部一二三区| 成人30分钟免费视频| 五十路熟女av天堂| 亚洲一区二区人妻av| 国产美女一区在线观看| 欧美日韩在线精品一区二区三| 成人午夜电影在线观看 久久| 激情人妻校园春色亚洲欧美| 风流唐伯虎电视剧在线观看 | 激情人妻校园春色亚洲欧美| 粉嫩av懂色av蜜臀av| 38av一区二区三区| 开心 色 六月 婷婷| 超碰97免费人妻麻豆| 欧美精品资源在线观看| 一区二区三区久久久91| 久久久麻豆精亚洲av麻花| 伊人开心婷婷国产av| 免费黄页网站4188| 亚洲综合图片20p| 亚洲精品亚洲人成在线导航| 绝顶痉挛大潮喷高潮无码 | 日韩美在线观看视频黄| 大鸡吧插逼逼视频免费看| 亚洲一区久久免费视频| 自拍偷拍一区二区三区图片| 91综合久久亚洲综合| 免费看美女脱光衣服的视频| 国产精品人妻66p| 国产无遮挡裸体免费直播视频| 97瑟瑟超碰在线香蕉| wwwxxx一级黄色片| 亚洲av黄色在线网站| 91av中文视频在线| 色在线观看视频免费的| 9国产精品久久久久老师| 熟女国产一区亚洲中文字幕| 91极品新人『兔兔』精品新作| 天美传媒mv视频在线观看| 精品日产卡一卡二卡国色天香| 国产chinesehd精品麻豆| 国产精品污污污久久| 国产极品精品免费视频| 91啪国自产中文字幕在线| 一色桃子久久精品亚洲 | 亚国产成人精品久久久| 成人伊人精品色xxxx视频| 美女大bxxxx内射| 国产又色又刺激在线视频| 男人插女人视频网站| 国产精品一区二区av国| 久久久久久性虐视频| 高清成人av一区三区| 久久精品美女免费视频| 一色桃子久久精品亚洲| 日韩av免费观看一区| 成人网18免费视频版国产| 男人天堂色男人av| 国产性生活中老年人视频网站| av日韩在线观看大全| 国产无遮挡裸体免费直播视频| 日韩美av高清在线| 日本xx片在线观看| 日视频免费在线观看| 初美沙希中文字幕在线| av大全在线播放免费| 欧洲日韩亚洲一区二区三区| 中国老熟女偷拍第一页| 国产夫妻视频在线观看免费| 天天做天天干天天操天天射| 99热久久这里只有精品8| 伊人综合aⅴ在线网| 日本少妇精品免费视频| 天天日天天敢天天干| 中文字幕无码一区二区免费| 人妻在线精品录音叫床| 91社福利《在线观看| 亚洲福利天堂久久久久久| 蜜臀av久久久久久久| 亚洲午夜电影在线观看| 91she九色精品国产| 欧美男人大鸡吧插女人视频| 久久免看30视频口爆视频| 蜜桃臀av蜜桃臀av| 特级无码毛片免费视频播放 | 亚洲综合一区二区精品久久| 免费高清自慰一区二区三区网站 | 40道精品招牌菜特色| 天天干天天插天天谢| 69精品视频一区二区在线观看| 爱有来生高清在线中文字幕| 99精品视频之69精品视频| 成人性爱在线看四区| 国产精彩对白一区二区三区| 天天操天天爽天天干| 91精品高清一区二区三区| 成人伊人精品色xxxx视频| 欲满人妻中文字幕在线| 欧美viboss性丰满| 男生舔女生逼逼视频| 一区二区三区四区五区性感视频| 午夜毛片不卡免费观看视频| 五月天久久激情视频| 免费看国产又粗又猛又爽又黄视频| 欧美激情电影免费在线| 视频 一区二区在线观看| 亚洲公开视频在线观看| 熟妇一区二区三区高清版| 精品一区二区三四区| av完全免费在线观看av| 最新日韩av传媒在线| 一级黄片久久久久久久久| 欧美香蕉人妻精品一区二区| 国产1区,2区,3区| av中文字幕网址在线| 欧美偷拍自拍色图片| 一本久久精品一区二区| 亚洲欧美人精品高清| 任我爽精品视频在线播放| 欧美一区二区三区在线资源| 国产成人精品av网站| 国产综合视频在线看片| 最新黄色av网站在线观看| 亚洲精品无码色午夜福利理论片| 91久久国产成人免费网站| 欧美国品一二三产区区别| 新97超碰在线观看| 国产无遮挡裸体免费直播视频| 久久精品亚洲成在人线a| 欧美交性又色又爽又黄麻豆| 水蜜桃一区二区三区在线观看视频| 在线免费观看欧美小视频| 青青青青青青青青青青草青青| 天天爽夜夜爽人人爽QC| 自拍偷区二区三区麻豆| 啊慢点鸡巴太大了啊舒服视频| 播放日本一区二区三区电影| 在线免费观看99视频| 日韩欧美国产一区不卡| 人妻少妇亚洲一区二区| 婷婷六月天中文字幕| 偷拍自拍视频图片免费| 天天操夜夜操天天操天天操 | 伊人成人综合开心网| 国产精选一区在线播放| 国产黑丝高跟鞋视频在线播放| 一级黄色av在线观看| 免费手机黄页网址大全| 亚洲av琪琪男人的天堂| 涩涩的视频在线观看视频| 视频一区 二区 三区 综合| 日韩三级黄色片网站| 国产在线观看黄色视频| 亚洲成人国产综合一区| 男女第一次视频在线观看| 青青擦在线视频国产在线| 午夜影院在线观看视频羞羞羞| 免费啪啪啪在线观看视频| 91天堂精品一区二区| 国产精品黄大片在线播放| 亚洲午夜精品小视频| 99久久99久国产黄毛片| 一区二区三区日韩久久| 在线观看国产网站资源| 在线国产精品一区二区三区| 天天日天天爽天天干| 日韩中文字幕在线播放第二页| 成人资源在线观看免费官网| 亚洲中文精品字幕在线观看| 美女大bxxxx内射| 亚洲专区激情在线观看视频| 精品亚洲国产中文自在线| 欧美老妇精品另类不卡片| 午夜激情久久不卡一区二区| 沙月文乃人妻侵犯中文字幕在线 | 精品国产乱码一区二区三区乱| 天堂av狠狠操蜜桃| 99精品视频之69精品视频| 日韩av有码中文字幕| 中文字幕在线视频一区二区三区| 男人天堂最新地址av| 99精品国产aⅴ在线观看| 亚洲日产av一区二区在线| 这里只有精品双飞在线播放| 日韩av大胆在线观看| 999热精品视频在线| 开心 色 六月 婷婷| 91国偷自产一区二区三区精品| 老有所依在线观看完整版 | 天天操天天干天天艹| 玖玖一区二区在线观看| 天码人妻一区二区三区在线看| 欧美成人黄片一区二区三区 | 亚洲1区2区3区精华液| 十八禁在线观看地址免费| 早川濑里奈av黑人番号| 亚洲另类在线免费观看| 天天日夜夜干天天操| 亚洲另类在线免费观看| 少妇高潮无套内谢麻豆| 欧美特级特黄a大片免费| 国产成人精品av网站| 绝顶痉挛大潮喷高潮无码| 人妻少妇亚洲一区二区| 国产女人叫床高潮大片视频| 国产欧美日韩第三页| 亚洲欧美综合另类13p| 精品亚洲国产中文自在线| 内射久久久久综合网| 阴茎插到阴道里面的视频| 中文字幕一区二区三区蜜月| 水蜜桃一区二区三区在线观看视频 | 亚洲日本一区二区久久久精品| 欧美一级视频一区二区| 亚洲欧美精品综合图片小说| 国产精品黄色的av| 青青青青青操视频在线观看| 国产成人精品av网站| 欧美成人黄片一区二区三区 | 亚洲自拍偷拍综合色| 女蜜桃臀紧身瑜伽裤| 第一福利视频在线观看| 国产成人午夜精品福利| 91极品大一女神正在播放| 青青青青青免费视频| 99久久久无码国产精品性出奶水 | 日辽宁老肥女在线观看视频| 亚洲一区二区三区在线高清| 97精品综合久久在线| av乱码一区二区三区| 白嫩白嫩美女极品国产在线观看| 国产精品人久久久久久| 人妻久久久精品69系列| 国产男女视频在线播放| 久久久久久9999久久久久| 欧美熟妇一区二区三区仙踪林| 亚洲一级特黄特黄黄色录像片| 91亚洲精品干熟女蜜桃频道| 成人H精品动漫在线无码播放| 97人妻人人澡爽人人精品| 在线免费观看黄页视频| 91天堂天天日天天操| 精品国产成人亚洲午夜| 中文字幕一区二区人妻电影冢本| 绝色少妇高潮3在线观看| 在线视频这里只有精品自拍| av在线播放国产不卡| 免费看国产av网站| 欧美黑人与人妻精品| 久草视频首页在线观看| 日本高清成人一区二区三区| 精品黑人一区二区三区久久国产 | 天堂av中文在线最新版| 亚洲国际青青操综合网站| 视频久久久久久久人妻| 全国亚洲男人的天堂| 国产精品精品精品999| 五十路熟女av天堂| 日本一二三中文字幕| 偷拍3456eee| 成人24小时免费视频| 亚洲av男人的天堂你懂的| 国产精品国产三级麻豆| 99精品国产自在现线观看| 久久综合老鸭窝色综合久久| 亚洲成av人无码不卡影片一| 成人H精品动漫在线无码播放| 在线免费观看日本片| 黄色的网站在线免费看| 日本午夜爽爽爽爽爽视频在线观看| 狠狠躁狠狠爱网站视频| 熟女人妻在线中出观看完整版| 国产在线免费观看成人| 熟女少妇激情五十路| 免费观看理论片完整版| 十八禁在线观看地址免费| 一区二区三区综合视频| 91精品激情五月婷婷在线| 黑人解禁人妻叶爱071| 少妇高潮一区二区三区| 在线免费观看日本片| 91国产资源在线视频| 天天日天天干天天爱| 天天插天天狠天天操| 色天天天天射天天舔| yellow在线播放av啊啊啊| 久久久久久久99精品| 国产一区二区火爆视频| 日本少妇人妻xxxxxhd| 日本真人性生活视频免费看| 成人网18免费视频版国产| 欧美精品国产综合久久| 午夜精品一区二区三区4| 动漫黑丝美女的鸡巴| 福利视频一区二区三区筱慧| 日本丰满熟妇BBXBBXHD| 久久免看30视频口爆视频| 黑人3p华裔熟女普通话| 亚洲一级av大片免费观看| 亚洲在线一区二区欧美| 青青青青在线视频免费观看| 中文字幕一区二区自拍| 欧亚日韩一区二区三区观看视频| 国产精品视频资源在线播放| 午夜精品福利91av| 一区二区三区四区视频在线播放| 久久午夜夜伦痒痒想咳嗽P| 91九色porny蝌蚪国产成人| 日本最新一二三区不卡在线| 91啪国自产中文字幕在线| 精品成人午夜免费看| 天天射夜夜操综合网| 99精品一区二区三区的区| 人妻丝袜诱惑我操她视频| 中文字幕在线乱码一区二区 | 91天堂精品一区二区| 99热久久这里只有精品8| 亚洲变态另类色图天堂网| 成人伊人精品色xxxx视频| av亚洲中文天堂字幕网| 亚洲精品ww久久久久久| 成人亚洲国产综合精品| 亚洲欧美一区二区三区爱爱动图 | 91传媒一区二区三区| 91在线免费观看成人| 日韩三级黄色片网站| 午夜精品福利91av| 免费看国产av网站| 亚洲av日韩精品久久久久久hd| 亚洲av香蕉一区区二区三区犇| 中文字幕高清在线免费播放 | 2018最新中文字幕在线观看| 亚洲中文字幕人妻一区| 蜜桃臀av蜜桃臀av| 日韩伦理短片在线观看| av天堂中文字幕最新| 小泽玛利亚视频在线观看| 国产女人被做到高潮免费视频| 日本性感美女三级视频| 91老熟女连续高潮对白| 日本又色又爽又黄又粗| caoporn蜜桃视频| 蜜臀av久久久久蜜臀av麻豆| 亚洲2021av天堂| 亚洲av天堂在线播放| 中文字幕 亚洲av| av男人天堂狠狠干| 一区二区三区另类在线| 丝袜肉丝一区二区三区四区在线 | 中文字幕网站你懂的| 91九色porny国产蝌蚪视频| 一区二区三区久久久91| 欧美一区二区三区啪啪同性| 任你操视频免费在线观看| 国产精品久久久久网| 久久久久久久99精品| 人妻少妇av在线观看| 91国产在线视频免费观看| 77久久久久国产精产品| 91精品资源免费观看| 激情人妻校园春色亚洲欧美| 亚洲欧美清纯唯美另类| 国产又粗又猛又爽又黄的视频美国| 中文字幕一区二区三区蜜月| 欧美特级特黄a大片免费| huangse网站在线观看| 欧美伊人久久大香线蕉综合| 日本性感美女写真视频| 综合一区二区三区蜜臀| 爱爱免费在线观看视频| 93精品视频在线观看| 激情人妻校园春色亚洲欧美| 国产免费av一区二区凹凸四季| 黑人借宿ntr人妻的沦陷2| 天天艹天天干天天操| 爆乳骚货内射骚货内射在线| av欧美网站在线观看| 亚洲国产免费av一区二区三区 | 欧美亚洲国产成人免费在线 | 2020国产在线不卡视频| 久久香蕉国产免费天天| 亚洲一级美女啪啪啪| 岛国av高清在线成人在线| 天天射,天天操,天天说| 经典亚洲伊人第一页| 大香蕉伊人国产在线| 一区二区视频在线观看视频在线| 国产91久久精品一区二区字幕| 日韩人妻丝袜中文字幕| 青青青青操在线观看免费| 传媒在线播放国产精品一区| 欧美日韩激情啪啪啪| 免费在线观看视频啪啪 | 欧美成人一二三在线网| 国产密臀av一区二区三| 国产麻豆国语对白露脸剧情| 亚洲1卡2卡三卡4卡在线观看 | 新97超碰在线观看| 中文字幕+中文字幕| 最新激情中文字幕视频| 亚洲精品久久视频婷婷| 天天日天天日天天擦| 久久久久久性虐视频| 激情五月婷婷综合色啪| 国产精品免费不卡av| 人妻少妇av在线观看| 亚洲天堂有码中文字幕视频| 久久久久五月天丁香社区| 亚洲国产免费av一区二区三区| 亚洲女人的天堂av| 国产又色又刺激在线视频| 天天干天天操天天摸天天射| 天天做天天干天天舔| aⅴ精产国品一二三产品| 亚洲av色香蕉一区二区三区| 欧美爆乳肉感大码在线观看 | 亚洲特黄aaaa片| 亚洲av成人网在线观看| 午夜精品久久久久久99热| 天天日天天做天天日天天做| av一本二本在线观看| 欧美3p在线观看一区二区三区| 福利午夜视频在线合集| 精品人人人妻人人玩日产欧| 9久在线视频只有精品| 亚洲成人线上免费视频观看| 激情综合治理六月婷婷| 国产熟妇乱妇熟色T区| 天天日天天透天天操| 亚洲精品中文字幕下载| 欧洲亚洲欧美日韩综合| 2018最新中文字幕在线观看| 国产午夜激情福利小视频在线| 在线播放一区二区三区Av无码| sw137 中文字幕 在线| 2020中文字幕在线播放| 成人色综合中文字幕| 日韩成人综艺在线播放| 中文字幕在线乱码一区二区| 天堂女人av一区二区| 少妇深喉口爆吞精韩国| 日本欧美视频在线观看三区| 国产欧美日韩第三页| 精品乱子伦一区二区三区免费播| 人妻少妇av在线观看| 国产揄拍高清国内精品对白| 日本一本午夜在线播放| huangse网站在线观看| 青青草原色片网站在线观看| gogo国模私拍视频| 粗大的内捧猛烈进出爽大牛汉子| 亚洲狠狠婷婷综合久久app | 亚洲av日韩高清hd| 神马午夜在线观看视频| 熟女人妻在线中出观看完整版| 久久久麻豆精亚洲av麻花| 久久99久久99精品影院| 亚洲欧美清纯唯美另类| 国产va精品免费观看| 日韩人妻在线视频免费| brazzers欧熟精品系列| 女生自摸在线观看一区二区三区| 护士特殊服务久久久久久久| 色婷婷综合激情五月免费观看| 激情啪啪啪啪一区二区三区 | 欧美久久久久久三级网| 欧美精产国品一二三产品区别大吗| 欧美黑人巨大性xxxxx猛交| av高潮迭起在线观看| 欧美色婷婷综合在线| 小穴多水久久精品免费看| mm131美女午夜爽爽爽| 搡老妇人老女人老熟女| 最新的中文字幕 亚洲| yellow在线播放av啊啊啊 | 国产精品三级三级三级| 99热这里只有精品中文| 九色视频在线观看免费| 毛片一级完整版免费| 黄色成年网站午夜在线观看| 亚洲1卡2卡三卡4卡在线观看| 成人资源在线观看免费官网| 亚洲av男人天堂久久| 日本女人一级免费片| 亚洲国产精品免费在线观看| 最新国产精品拍在线观看| 偷拍美女一区二区三区| 超碰中文字幕免费观看| 女生自摸在线观看一区二区三区| 久久久久久9999久久久久| 免费无毒热热热热热热久| 日韩人妻xxxxx| 天天射夜夜操综合网| 专门看国产熟妇的网站| 操人妻嗷嗷叫视频一区二区 | 国产激情av网站在线观看| gay gay男男瑟瑟在线网站| 91香蕉成人app下载| 日本人妻欲求不满中文字幕| 日韩近亲视频在线观看| 国产成人自拍视频在线免费观看| 亚洲人人妻一区二区三区| 亚洲综合另类精品小说| 亚洲熟妇无码一区二区三区| 中文字幕乱码人妻电影| 丝袜美腿欧美另类 中文字幕| 亚洲av人人澡人人爽人人爱| 人妻久久无码中文成人| 老司机福利精品视频在线| 红杏久久av人妻一区| 亚洲综合一区成人在线| 欧美成一区二区三区四区| 欧美亚洲牲夜夜综合久久| 日韩在线中文字幕色| 欧亚日韩一区二区三区观看视频| 国产精品亚洲а∨天堂免| 亚洲欧美一卡二卡三卡| 男人操女人逼逼视频网站| 国产日韩精品一二三区久久久| 2022国产综合在线干| 97国产福利小视频合集| 国产97视频在线精品| 中文字幕熟女人妻久久久| 日本韩国亚洲综合日韩欧美国产| 成熟熟女国产精品一区| 亚洲综合在线视频可播放| 亚洲欧洲av天堂综合| 欧美另类重口味极品在线观看| 成年人该看的视频黄免费| 香港一级特黄大片在线播放| 亚洲免费va在线播放| 欧美一区二区三区在线资源| 91传媒一区二区三区| 91人妻精品久久久久久久网站 | 国产又粗又硬又大视频| av一本二本在线观看| 3337p日本欧洲大胆色噜噜| 懂色av之国产精品| av在线观看网址av| avjpm亚洲伊人久久| 日韩美女综合中文字幕pp| 中文字幕日韩91人妻在线| 中文字幕日韩精品日本| 欧美一级色视频美日韩| 蜜桃久久久久久久人妻| 久久久久久cao我的性感人妻| 国产麻豆剧传媒精品国产av蜜桃| 40道精品招牌菜特色| 亚洲最大免费在线观看| 色噜噜噜噜18禁止观看| 天天日夜夜操天天摸| 亚洲天天干 夜夜操| 人人在线视频一区二区| 亚洲欧美久久久久久久久| 亚洲另类在线免费观看| 色婷婷六月亚洲综合香蕉| 中文乱理伦片在线观看| 老司机午夜精品视频资源| 美女吃鸡巴操逼高潮视频| 亚洲av人人澡人人爽人人爱| 99精品国产免费久久| 日韩人妻在线视频免费| 日本人妻少妇18—xx| 国产精品探花熟女在线观看| 91快播视频在线观看| 午夜精品久久久久久99热| 超碰中文字幕免费观看| 国产高清在线观看1区2区| 亚洲最大黄 嗯色 操 啊| 亚洲色偷偷综合亚洲AV伊人| 综合激情网激情五月五月婷婷| 国产极品美女久久久久久| 日韩欧美一级精品在线观看| 亚洲欧美色一区二区| 天天干天天操天天玩天天射| 免费国产性生活视频| 亚洲精品高清自拍av| 国产日韩一区二区在线看| 久草电影免费在线观看| jul—619中文字幕在线| 亚洲欧美综合在线探花| 人妻久久久精品69系列| 亚洲欧美在线视频第一页| 婷婷色中文亚洲网68| 综合国产成人在线观看| 日韩激情文学在线视频| 国产精品一区二区av国| 丝袜亚洲另类欧美变态| 中文字幕日韩精品就在这里| 在线 中文字幕 一区| 97a片免费在线观看| 最近中文字幕国产在线| 日韩精品啪啪视频一道免费| 国产福利小视频免费观看| 欧美熟妇一区二区三区仙踪林| 2018最新中文字幕在线观看| 99久久久无码国产精品性出奶水| 久久久久久九九99精品| av在线观看网址av| 啊啊好大好爽啊啊操我啊啊视频| 大鸡吧插逼逼视频免费看| 老师啊太大了啊啊啊尻视频| 欧美熟妇一区二区三区仙踪林| 日本熟妇喷水xxx| 久久精品亚洲国产av香蕉| 美女大bxxxx内射| 中国无遮挡白丝袜二区精品| 国产亚洲欧美视频网站| 一级黄片久久久久久久久| 绝顶痉挛大潮喷高潮无码 | 99精品视频在线观看免费播放| 沈阳熟妇28厘米大战黑人| 福利一二三在线视频观看| 精品一区二区亚洲欧美| 欧美性感尤物人妻在线免费看| 一区二区三区国产精选在线播放 | 老师让我插进去69AV| 色秀欧美视频第一页| 欧美日韩激情啪啪啪| 在线观看成人国产电影| 天天日天天干天天搡| 狠狠的往里顶撞h百合| 97香蕉碰碰人妻国产樱花| 亚洲va国产va欧美精品88| 香蕉91一区二区三区| 老司机福利精品免费视频一区二区| 国产亚洲视频在线观看| 97人人模人人爽人人喊| 久草视频在线免播放| 久草视频 久草视频2| 视频啪啪啪免费观看| 淫秽激情视频免费观看| 在线观看视频污一区| 黄色片一级美女黄色片| 久久精品视频一区二区三区四区| 国产亚洲视频在线二区| av资源中文字幕在线观看| 欧美日本国产自视大全| 午夜精品一区二区三区福利视频| 欧美精产国品一二三区| 老司机免费福利视频网| 中文字幕亚洲久久久| 不卡精品视频在线观看| 成人免费毛片aaaa| 岛国青草视频在线观看| 在线新三级黄伊人网| 在线视频这里只有精品自拍| 51国产偷自视频在线播放| 日本性感美女三级视频| 亚洲av黄色在线网站| 亚洲男人让女人爽的视频| 亚洲国产成人无码麻豆艾秋| 一区二区三区激情在线| 精品美女在线观看视频在线观看 | 国产精彩福利精品视频| 99久久成人日韩欧美精品| 绝色少妇高潮3在线观看| 青青青激情在线观看视频| 99精品视频在线观看婷婷| 99热久久极品热亚洲| 91九色国产熟女一区二区| 国际av大片在线免费观看| av天堂加勒比在线| 午夜美女少妇福利视频| www骚国产精品视频| 伊人精品福利综合导航| 精品av国产一区二区三区四区| 91欧美在线免费观看| 三级av中文字幕在线观看| 国产刺激激情美女网站| 亚洲av极品精品在线观看| 欧美日韩熟女一区二区三区| 我想看操逼黄色大片| weyvv5国产成人精品的视频| 91久久综合男人天堂| 美女福利写真在线观看视频| 日本免费午夜视频网站| 岛国一区二区三区视频在线| 热99re69精品8在线播放| av在线资源中文字幕| 青青青青青青青青青青草青青 | 91色老99久久九九爱精品| 中文字幕1卡1区2区3区| 最新国产亚洲精品中文在线| 欧美中国日韩久久精品| 青草青永久在线视频18| 内射久久久久综合网| 精品区一区二区三区四区人妻| 动色av一区二区三区| 午夜激情久久不卡一区二区| 欧美亚洲偷拍自拍色图| 欧美少妇性一区二区三区| 福利一二三在线视频观看| 中国熟女@视频91| 黄色中文字幕在线播放| 特黄老太婆aa毛毛片| 99国内精品永久免费视频| 青青草视频手机免费在线观看| 一区二区在线观看少妇| 青青尤物在线观看视频网站| 老有所依在线观看完整版| 欧美精产国品一二三产品区别大吗| 美女张开两腿让男人桶av| 操操网操操伊剧情片中文字幕网| 91一区精品在线观看| 青青青青操在线观看免费| 人妻最新视频在线免费观看| 一区二区三区激情在线| 69精品视频一区二区在线观看| 在线免费观看99视频| 水蜜桃国产一区二区三区| 久草免费人妻视频在线| 国产乱子伦精品视频潮优女| 日日摸夜夜添夜夜添毛片性色av| 精品少妇一二三视频在线| 人妻丝袜精品中文字幕| 欧美精产国品一二三区| 欧美专区第八页一区在线播放 | 无码国产精品一区二区高潮久久4| 天天日天天透天天操| 操的小逼流水的文章| 免费一级特黄特色大片在线观看| 日韩中文字幕精品淫| 黄网十四区丁香社区激情五月天 | 色av色婷婷人妻久久久精品高清 | 成人午夜电影在线观看 久久| 2022国产精品视频| 黄色av网站免费在线| 在线新三级黄伊人网| 成年午夜影片国产片| 9国产精品久久久久老师| 亚洲欧美另类手机在线| 亚洲一级av大片免费观看| 好吊操视频这里只有精品| 中文字幕在线第一页成人| 男人操女人逼逼视频网站| 男人在床上插女人视频| 国产露脸对白在线观看| 白嫩白嫩美女极品国产在线观看| 又色又爽又黄又刺激av网站| 青青青青青免费视频| 18禁精品网站久久| 日韩精品电影亚洲一区| 年轻的人妻被夫上司侵犯| 亚洲va天堂va国产va久| 人妻爱爱 中文字幕| 狠狠操狠狠操免费视频| 馒头大胆亚洲一区二区| 91精品国产综合久久久蜜| 精彩视频99免费在线| 天天色天天操天天透| 亚洲欧美综合在线探花| 日本一区美女福利视频| 欧美在线一二三视频| 国产无遮挡裸体免费直播视频| 午夜激情精品福利视频| 中文字幕第1页av一天堂网| 国产日韩一区二区在线看| h国产小视频福利在线观看| 国产97在线视频观看| 精品视频国产在线观看| 亚洲欧美久久久久久久久| 国产精选一区在线播放| 自拍偷拍vs一区二区三区| 天天色天天操天天舔| 大陆av手机在线观看| 国产精品国产三级麻豆| 大鸡巴操b视频在线| 沙月文乃人妻侵犯中文字幕在线| 日本韩国免费一区二区三区视频| 精品一区二区三四区| 国产亚洲天堂天天一区| 99久久成人日韩欧美精品| 天堂av狠狠操蜜桃| 国产日韩av一区二区在线| 97精品人妻一区二区三区精品| 天天干天天操天天扣| 中文字幕亚洲久久久| 啊啊啊想要被插进去视频| 强行扒开双腿猛烈进入免费版| 伊人日日日草夜夜草| 亚洲日本一区二区久久久精品| 亚洲青青操骚货在线视频| 家庭女教师中文字幕在线播放| 天天日天天敢天天干| 大香蕉大香蕉在线看| 九九视频在线精品播放| 天干天天天色天天日天天射| 亚洲成高清a人片在线观看| 偷拍美女一区二区三区| 亚洲伊人色一综合网| 一区二区三区精品日本| jul—619中文字幕在线| av线天堂在线观看| 国产视频精品资源网站| 姐姐的朋友2在线观看中文字幕 | 在线免费观看日本片| 亚洲第一伊人天堂网| 久久久极品久久蜜桃| 色综合久久五月色婷婷综合| 亚洲少妇人妻无码精品| 成年人啪啪视频在线观看| 色av色婷婷人妻久久久精品高清| 在线观看的a站 最新| 日本免费视频午夜福利视频| 国产免费av一区二区凹凸四季| 夜女神免费福利视频| 新婚人妻聚会被中出| 国产精品人妻一区二区三区网站| 韩国黄色一级二级三级| 岛国青草视频在线观看| 任你操视频免费在线观看| 精品国产亚洲av一淫| 啪啪啪啪啪啪啪啪av| 亚洲综合另类精品小说| 乱亲女秽乱长久久久| 欧美国产亚洲中英文字幕| 亚洲av日韩av第一区二区三区| 色伦色伦777国产精品| 午夜青青草原网在线观看| 最新激情中文字幕视频| 99久久99久国产黄毛片| 搡老熟女一区二区在线观看| av亚洲中文天堂字幕网| 青青青青操在线观看免费| 黄色片黄色片wyaa| 免费看高清av的网站| 91精品资源免费观看| 岛国一区二区三区视频在线| 国产第一美女一区二区三区四区| 亚洲人妻30pwc| 国产麻豆乱子伦午夜视频观看| 亚洲一区二区三区久久午夜 | 婷婷五月亚洲综合在线| 亚洲欧美激情中文字幕| 亚洲精品一线二线在线观看| 日本高清在线不卡一区二区| 日韩三级电影华丽的外出 | 人人妻人人人操人人人爽| 4个黑人操素人视频网站精品91| 美日韩在线视频免费看| 91欧美在线免费观看| 国产黄色片蝌蚪九色91| 人妻少妇性色欲欧美日韩| 99婷婷在线观看视频| 亚洲成a人片777777| 亚洲精品国产久久久久久| 晚上一个人看操B片| 污污小视频91在线观看| 成人av电影免费版| 秋霞午夜av福利经典影视| 91国产资源在线视频| 黄网十四区丁香社区激情五月天| 中文字幕奴隷色的舞台50| av天堂加勒比在线| 六月婷婷激情一区二区三区| 免费男阳茎伸入女阳道视频 | 在线观看成人国产电影| 男女啪啪啪啪啪的网站| 国产熟妇一区二区三区av | 女生自摸在线观看一区二区三区 | 日韩午夜福利精品试看| 2021最新热播中文字幕| 啊啊好慢点插舔我逼啊啊啊视频| 亚洲美女高潮喷浆视频| 99精品视频在线观看婷婷| 天天日天天玩天天摸| avjpm亚洲伊人久久| 免费在线播放a级片| 非洲黑人一级特黄片| 小穴多水久久精品免费看| 五月精品丁香久久久久福利社| 亚洲精品高清自拍av | 日噜噜噜夜夜噜噜噜天天噜噜噜| 国产欧美日韩在线观看不卡| 亚洲欧美精品综合图片小说| 免费国产性生活视频| 丰满的继坶3中文在线观看| 美女吃鸡巴操逼高潮视频| 国产+亚洲+欧美+另类| 91精品啪在线免费| 亚洲va天堂va国产va久| 在线观看日韩激情视频| 欧美成一区二区三区四区| 日韩影片一区二区三区不卡免费| 午夜婷婷在线观看视频| 天堂女人av一区二区| 亚洲国产香蕉视频在线播放| 好吊视频—区二区三区| 亚洲在线观看中文字幕av| 岛国免费大片在线观看| 中文字幕中文字幕 亚洲国产| 久草免费人妻视频在线| 人妻av无码专区久久绿巨人| 人妻自拍视频中国大陆| 日本韩国免费一区二区三区视频| 亚洲av无乱一区二区三区性色| 蜜桃专区一区二区在线观看| 国产揄拍高清国内精品对白| 国产精品自偷自拍啪啪啪| 福利视频网久久91| 日韩精品中文字幕播放| 日本后入视频在线观看| 日韩加勒比东京热二区| 最新91九色国产在线观看| 成人30分钟免费视频| 免费在线看的黄片视频| 欧美精品一区二区三区xxxx| 日本最新一二三区不卡在线 | 亚洲av无硬久久精品蜜桃| 中文字幕在线永久免费播放| 亚洲高清国产拍青青草原| www日韩毛片av| 欧美精品中文字幕久久二区| 99精品视频在线观看婷婷| 天天干天天搞天天摸| 欧美女同性恋免费a| 亚洲精品午夜久久久久| 欧美日韩精品永久免费网址| 青青尤物在线观看视频网站| 51精品视频免费在线观看| 亚洲欧美激情国产综合久久久| 国产在线观看黄色视频| 久久久久久性虐视频| 天堂av在线播放免费| 天天做天天爽夜夜做少妇| 操操网操操伊剧情片中文字幕网| 91国内视频在线观看| 亚洲天堂精品福利成人av| 18禁美女黄网站色大片下载| av手机在线免费观看日韩av| 亚洲熟妇久久无码精品| 2020久久躁狠狠躁夜夜躁 | 欧美特级特黄a大片免费| 天天日夜夜干天天操| 久草视频首页在线观看| 日本美女成人在线视频| 亚洲中文字幕乱码区| 国产成人自拍视频在线免费观看| 成年女人免费播放视频| 久久久久久九九99精品| 成人网18免费视频版国产| 无码中文字幕波多野不卡| 日韩欧美在线观看不卡一区二区 | 国产精品久久久久国产三级试频| 人妻另类专区欧美制服| 亚洲精品ww久久久久久| 国产午夜亚洲精品麻豆| 久久久久久久亚洲午夜综合福利| 日韩av有码中文字幕| 自拍 日韩 欧美激情| 国产精品亚洲在线观看| 亚洲精品在线资源站| 日韩精品激情在线观看| 成熟丰满熟妇高潮xx×xx| 五十路熟女人妻一区二区9933 | 亚洲第17页国产精品| 久草视频在线一区二区三区资源站| 特大黑人巨大xxxx| 91天堂天天日天天操| sejizz在线视频| 日本少妇精品免费视频| 日比视频老公慢点好舒服啊| 国产变态另类在线观看| 亚洲高清自偷揄拍自拍| 91麻豆精品91久久久久同性| 超碰97人人澡人人| 欧美aa一级一区三区四区| 亚洲免费成人a v| 亚洲欧美另类自拍偷拍色图| 污污小视频91在线观看| 女同久久精品秋霞网| 一区二区在线视频中文字幕| 一区二区久久成人网| 熟女人妻在线中出观看完整版| 日日爽天天干夜夜操| 中文字幕日韩人妻在线三区| 国产一区二区三免费视频| 日日夜夜精品一二三| 最近中文2019年在线看| 精品美女久久久久久| 欧美黑人性暴力猛交喷水| 亚洲va天堂va国产va久| 日韩av有码中文字幕| 姐姐的朋友2在线观看中文字幕| aiss午夜免费视频| 亚洲的电影一区二区三区| 亚洲高清国产一区二区三区| 大白屁股精品视频国产| 久久免看30视频口爆视频| 夜色福利视频在线观看| 欧美精产国品一二三产品价格| 久草视频在线看免费| 神马午夜在线观看视频| 日本人妻精品久久久久久| 一区二区三区四区五区性感视频|