spring cloud alibaba 实现将外部项目注册到nacos+健康检查

pom 依赖

        <gson.version>2.10</gson.version>
        <fjson.version>1.2.15</fjson.version>
        <hutool.version>5.8.11</hutool.version>
        <apcommon.version>3.1</apcommon.version>
        <google.guava.version>30.1-jre</google.guava.version>
        <spring.mvc.version>5.2.7.RELEASE</spring.mvc.version>
        <jasypt-spring-boot-starter.version>3.0.4</jasypt-spring-boot-starter.version>

<dependencies>


        <!-- nacos 服务注册发现  -->
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
            <exclusions>
                <!-- 将ribbon排除 -->
                <exclusion>
                    <groupId>org.springframework.cloud</groupId>
                    <artifactId>spring-cloud-starter-netflix-ribbon</artifactId>
                </exclusion>
            </exclusions>
        </dependency>

        <!--添加loadbalancer依赖
            由于 Netflix Ribbon 进入停更维护阶段,因此 SpringCloud 2020.0.1 版本之后 删除了eureka中的ribbon,替代ribbon的是spring cloud自带的LoadBalancer,默认使用的是轮询的方式
            新版本的 Nacos discovery 都已经移除了 Ribbon ,此时我们需要引入 loadbalancer 代替,才能调用服务提供者提供的服务
        -->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-loadbalancer</artifactId>
        </dependency>
        <!--配置中心-->
      <!--  <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId>
        </dependency>-->

        <!-- 支持读取bootstrap配置 -->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-bootstrap</artifactId>
        </dependency>
        <!--encryption tool starts encryption-->
        <dependency>
            <groupId>com.github.ulisesbocchio</groupId>
            <artifactId>jasypt-spring-boot-starter</artifactId>
            <version>${jasypt-spring-boot-starter.version}</version>
        </dependency>
        <!--actuator-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-actuator</artifactId>

            <exclusions>
                <exclusion>
                    <groupId>org.springframework.boot</groupId>
                    <artifactId>spring-boot-starter-logging</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
        <!-- web依赖包-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>

            <!-- 去掉springboot默认配置 -->
            <exclusions>
                <exclusion>
                    <groupId>org.springframework.boot</groupId>
                    <artifactId>spring-boot-starter-logging</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
        <!-- 引入log4j2依赖 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-log4j2</artifactId>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>

        <!--fastjson-->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>${fjson.version}</version>
        </dependency>
        <!--guava-->
        <dependency>
            <groupId>com.google.guava</groupId>
            <artifactId>guava</artifactId>
            <version>${google.guava.version}</version>
        </dependency>
        <!--commons-lang3-->
        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-lang3</artifactId>
            <version>${apcommon.version}</version>
        </dependency>
        <!--hutool-->
        <dependency>
            <groupId>cn.hutool</groupId>
            <artifactId>hutool-all</artifactId>
            <version>${hutool.version}</version>
        </dependency>
        <!--dataformat-->
        <!-- 加上这个才能辨认到log4j2.yml文件 -->
        <dependency>
            <groupId>com.fasterxml.jackson.dataformat</groupId>
            <artifactId>jackson-dataformat-yaml</artifactId>
        </dependency>

        <!-- 支持读取bootstrap配置 -->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-bootstrap</artifactId>
        </dependency>
<!--
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-consul-discovery</artifactId>
            <optional>true</optional>
        </dependency>-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-configuration-processor</artifactId>
            <optional>true</optional>
        </dependency>
    </dependencies>
    <dependencyManagement>
        <!--SpringBoot整合Spring Cloud-->
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>2021.0.4</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
            <!--SpringBoot整合Spring Cloud Alibaba-->
            <dependency>
                <groupId>com.alibaba.cloud</groupId>
                <artifactId>spring-cloud-alibaba-dependencies</artifactId>
                <version>2021.0.4.0</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

yml配置

spring:
  application:
    name: hlvy-nacos-regis
    description: 服务发现与注册服务
  cloud:
    nacos:
      discovery:
        server-addr: 你的nacos地址
        #指定命名空间 可以用于区分环境 只有同一环境下服务才可以调用
        namespace: 命名空间
        #指定集群名称
        cluster-name: ShangHai
        username: 账号
        password: 密码
server:
  port: 8849
management:
  endpoints:
    web:
      exposure:
        include: '*'
  endpoint:
    health:
      show-details: always
wii:
  # 异构微服务的IP,异构微服务的端口,注册服务名,分组默认DEFAULT_GROUP,异构微服务的健康检查URL可以不填则不检测(注意如果需要检测定义一个接口返回status: "UP"即可)
  # 127.0.0.1,8060,huf-user,DEFAULT_GROUP,http://localhost:8060/health.json
   regisService:
     - 127.0.0.0,8847,hlvy-test,DEFAULT_GROUP

代码:

WiiProperties

import lombok.Data;

import java.net.URI;
/**
 * @title: 服务注册信息
 * @author: luo heng
 * @date: 2023/2/28 16:44
 * @description: TODO
 * @version:
 * @return:
 */
@Data
public class WiiProperties {
    /**
     * polyglot service's ip
     */
    private String ip;

    /**
     * polyglot service's port
     */
    private Integer port;

    /**
     * 注册名
     */
    private String nacosName ;

    /**
     * 分组
     */
    private String group = "DEFAULT_GROUP";

    /**
     * polyglot service's health check url.
     * this endpoint must return json and the format must follow spring boot actuator's health endpoint.
     * eg. {"status": "UP"}
     */
    private URI healthCheckUrl;
}
ExWillProperties

import lombok.Data;
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.PropertySource;

import java.net.URI;
import java.util.ArrayList;
import java.util.List;

/**
 * @program: hyf-nacos-regis
 * @description: 需要注册的服务信息
 * @author: loren
 * @Description: TODO
 * @create: 2023-02-28 11:15
 **/
@Slf4j
@Data
@ConfigurationProperties(prefix = "wii")
@PropertySource(value = "classpath:application-${spring.profiles.active}.yml", encoding = "UTF-8")
public class ExWillProperties {

    private List<String> regisService;

    /**
     * 解析配置
     * @return
     */
    public List<WiiProperties> getWiiProperties(){
        List<WiiProperties> list = new ArrayList<>();
      try {
          WiiProperties wiiProperties = null;
          if (null != this.regisService){
              for (String info : regisService) {
                  String[] split = info.split(",");
                  int length = split.length;
                  if (length >= 3) {
                      wiiProperties = new WiiProperties();
                      wiiProperties.setIp(split[0]);
                      wiiProperties.setPort(Integer.valueOf(split[1]));
                      wiiProperties.setNacosName(split[2]);
                      if (length >= 4) {
                          wiiProperties.setGroup(split[3]);
                          if (length >= 5) {
                              wiiProperties.setHealthCheckUrl(URI.create(split[4]));
                          }
                      }
                      list.add(wiiProperties);
                  }

              }
          }

      }catch (Exception e){
          log.warn("wiiProperties error:{}",e.getMessage());
      }
        return list;
    }
}
WiiHealthIndicator

import lombok.RequiredArgsConstructor;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.actuate.health.Health;
import org.springframework.core.ParameterizedTypeReference;
import org.springframework.http.HttpMethod;
import org.springframework.http.ResponseEntity;
import org.springframework.web.client.RestTemplate;

import java.net.URI;
import java.util.Map;
/**
 * @title: 健康检查
 * @author: luo heng
 * @date: 2023/2/28 16:43
 * @description: TODO
 * @version:
 * @return:
 */
@RequiredArgsConstructor(onConstructor = @__(@Autowired))
public class WiiHealthIndicator   {
    private final RestTemplate restTemplate;

    public Health doHealthCheck(Health.Builder builder,URI healthCheckUrl) throws Exception {
        try {
            URI uri = healthCheckUrl;
            if (uri == null) {
                builder.up();
                return builder.build();
            }

            ResponseEntity<Map<String, Object>> exchange = this.restTemplate.exchange(
                    uri,
                    HttpMethod.GET,
                    null,
                    new ParameterizedTypeReference<Map<String, Object>>() {
                    }
            );

            Map<String, Object> map = exchange.getBody();

            if (map == null) {
                this.getWarning(builder);
                return builder.build();
            }
            Object status = map.get("status");
            if (status instanceof String) {
                builder.status(status.toString());
            } else {
                this.getWarning(builder);
            }
        } catch (Exception e) {
            builder.down().withDetail("error", e.getMessage());
        }
        return builder.build();
    }

    private void getWarning(Health.Builder builder) {
        builder.unknown().withDetail("warning", "no status field in response");
    }
}
WiiAutoConfiguration

import com.alibaba.cloud.nacos.NacosDiscoveryProperties;
import com.hy.hyfnacosregis.nacos.properties.ExWillProperties;
import com.hy.hyfnacosregis.nacos.wii.nacos.WiiChecker;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.client.RestTemplate;

/**
 * @title: 服务注册配置
 * @author: luo heng
 * @date: 2023/2/28 16:44
 * @description: TODO
 * @version:
 * @return:
 */
@Configuration
@EnableConfigurationProperties({ExWillProperties.class})
public class WiiAutoConfiguration {
    @Bean
    @ConditionalOnMissingBean
    public RestTemplate restTemplate() {
        return new RestTemplate();
    }

    @Bean
    public WiiHealthIndicator wiiHealthIndicator(RestTemplate restTemplate) {
        return new WiiHealthIndicator(restTemplate);
    }

    @Bean
    public WiiChecker wiiCleaner(NacosDiscoveryProperties nacosDiscoveryProperties, WiiHealthIndicator wiiHealthIndicator, ExWillProperties exWillProperties ) {
        WiiChecker cleaner = new WiiChecker(nacosDiscoveryProperties, wiiHealthIndicator,exWillProperties);
        cleaner.check();
        return cleaner;
    }
}
WiiChecker

import com.alibaba.cloud.nacos.NacosDiscoveryProperties;
import com.hy.hyfnacosregis.nacos.properties.ExWillProperties;
import com.hy.hyfnacosregis.nacos.properties.WiiProperties;
import com.hy.hyfnacosregis.nacos.wii.WiiHealthIndicator;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.actuate.health.Health;
import org.springframework.boot.actuate.health.Status;
import reactor.core.scheduler.Schedulers;

import java.util.List;
import java.util.concurrent.TimeUnit;

/**
 * @author www.itmuch.com
 */
@Slf4j
@RequiredArgsConstructor(onConstructor = @__(@Autowired))
public class WiiChecker {
    private final NacosDiscoveryProperties  nacosDiscoveryProperties;
    private final WiiHealthIndicator wiiHealthIndicator;
    private final ExWillProperties exWillProperties;

    /**
     * 健康检查
     */
    public void check() {

        Schedulers.single()
                .schedulePeriodically(
                        () -> {
                            try {
                            List<WiiProperties> wiiPropertiesList = exWillProperties.getWiiProperties();
                            if (null != wiiPropertiesList &&wiiPropertiesList.size() > 0){
                                for (WiiProperties properties : wiiPropertiesList) {
                                    String ip = properties.getIp();
                                    Integer port = properties.getPort();
                                    Health.Builder builder = new Health.Builder();
                                    Status status = new Status("DOWN");
                                    status = wiiHealthIndicator.doHealthCheck(builder,properties.getHealthCheckUrl()).getStatus();

                                    String applicationName = properties.getNacosName();
                                    if (status.equals(Status.UP)) {

                                            this.nacosDiscoveryProperties.namingServiceInstance().registerInstance(applicationName, ip, port);

                                        log.debug("Health check success. register this instance. applicationName = {}, ip = {}, port = {}, status = {}",
                                                applicationName, ip, port, status
                                        );
                                    } else {
                                        log.warn("Health check failed. unregister this instance. applicationName = {}, ip = {}, port = {}, status = {}",
                                                applicationName, ip, port, status
                                        );
                                            this.nacosDiscoveryProperties.namingServiceInstance().deregisterInstance(applicationName, ip, port);

                                    }
                                   // Thread.sleep(1000L);
                                }
                            }

                              } catch (Exception e) {
                               throw new RuntimeException(e);
                             }
                        },
                        0,
                        60,
                        TimeUnit.SECONDS
                );

    }
}

注意:目前暂未支持命名空间注册,所配置的服务都会注册到当前的命名空间的nacos中,分组也暂未实现 ,可根据当前代码扩展

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是 Spring Cloud Alibaba + Spring Boot + Nacos + Mybatis Plus + MySQL 项目搭建步骤: 1. 创建 Spring Boot 项目 使用 Spring Initializr 创建一个新的 Spring Boot 项目,选择需要的依赖,例如 Web、MySQL、Mybatis Plus 等。 2. 集成 Spring Cloud Alibaba 在 pom.xml 中添加 Spring Cloud Alibaba 的依赖: ``` <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-alibaba-dependencies</artifactId> <version>2.2.1.RELEASE</version> <type>pom</type> <scope>import</scope> </dependency> ``` 3. 集成 Nacos 在 pom.xml 中添加 Nacos 的依赖: ``` <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId> </dependency> ``` 在 application.properties 中添加 Nacos 的配置: ``` spring.cloud.nacos.discovery.server-addr=127.0.0.1:8848 ``` 4. 集成 Mybatis Plus 在 pom.xml 中添加 Mybatis Plus 的依赖: ``` <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter</artifactId> <version>3.4.0</version> </dependency> ``` 5. 集成 MySQL 在 pom.xml 中添加 MySQL 的依赖: ``` <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> </dependency> ``` 在 application.properties 中添加 MySQL 的配置: ``` spring.datasource.url=jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=utf-8&useSSL=false&serverTimezone=GMT%2B8 spring.datasource.username=root spring.datasource.password=root spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver ``` 6. 编写代码 在 src/main/java 目录下创建包和类,编写业务逻辑代码和数据访问代码。 7. 启动项目 运行项目,访问 http://localhost:8080 可以看到项目的首页。 以上是 Spring Cloud Alibaba + Spring Boot + Nacos + Mybatis Plus + MySQL 项目搭建步骤,希望能对你有所帮助。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值