eureka unknownHostException в обнаружении службы

У меня два микросервиса,

  1. eureka-client-1 работает на локальном хосте: 8081
  2. eureka-client-2 работает на локальном хосте: 8082

Оба они являются DiscoveryClients, зарегистрированными на «eureka-server», работающем на localhost: 8761.

Во фрагменте кода ниже я пытаюсь вызвать eureka-client-2 из eureka-client-1. Вместо вызова http://localhost:8082 я хочу вызвать http://eureka-client-2, но это вызывает исключение java.net.UnknownHostException во время обнаружения службы Eureka.

После поиска я обнаружил, что мне нужно использовать «Брикстон», чтобы это сделать.

Есть ли способ сделать это с помощью Camden.SR3?

Пожалуйста, предложите.

@Component
public class HystrixDemoService {   

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

    @HystrixCommand(fallbackMethod = "getFallbackCustomerName")
    public String getCustomerName() {
        RestTemplate restTemplate = new RestTemplate();
        URI uri = URI.create("http://eureka-client-2");     // fails here
        return restTemplate.getForObject(uri, String.class);
    }

    public String getFallbackCustomerName() {
        System.out.println("coming inside fallback method");
        return "Resillient Customer";
    }
}

pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.example</groupId>
    <artifactId>demo-pranay-eureka-client1</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>demo-pranay-eureka-client1</name>
    <description>Demo project for Spring Boot</description>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.4.2.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-actuator</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-eureka</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-hystrix</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>Camden.SR3</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>


</project>

application.properties для клиента 1, аналогично для клиента 2 (просто измените имя, т.е. eureka-client-2)

spring.application.name=eureka-client-1
server.port=8081
eureka:
  client:
    registerWithEureka: true
    serviceUrl:
      defaultZone: http://localhost:8761/eureka/
  instance:
    leaseRenewalIntervalInSeconds: 10
    statusPageUrlPath: /info
    healthCheckUrlPath: /health

application.properties для сервера eureka

spring.application.name=eureka-service
server.port=8761
eureka:
  client:
    registerWithEureka: false
    serviceUrl:
      defaultZone: http://localhost:8761/eureka/
  instance:
    leaseRenewalIntervalInSeconds: 10
    statusPageUrlPath: /info
    healthCheckUrlPath: /health

person prranay    schedule 19.12.2016    source источник
comment
Не могли бы вы показать свои файлы конфигурации (application.yml, application.properties, bootstrap.yml, ...)   -  person ootero    schedule 20.12.2016
comment
привет @ootero, обновил вопрос. application.properties файл.   -  person prranay    schedule 21.12.2016
comment
Когда вы открываете Eureka сервер по адресу: localhost: 8761 / eureka, можете ли вы увидеть eureka-client-1 а eureka-client-1 зарегистрирован? Я думаю, вам не хватает Ribbon конфигурации в eureka-client-1   -  person ootero    schedule 21.12.2016
comment
@ootero, я все еще пытаюсь понять это правильно. Не могли бы вы пройти через код github.com/pranayhere/SpringBootDemo. Это будет очень полезно.   -  person prranay    schedule 21.12.2016
comment
@ootero, я вижу, что eureka-client-1 и eureka-client-2 зарегистрированы на localhost: 8761. Согласно коду, когда я нажимаю localhost: 8081, он должен вернуть, что я говорю с B, но поскольку я получаю unknownHostException, это приводит к тому, что я говорю с устойчивым клиентом.   -  person prranay    schedule 21.12.2016
comment
Пожалуйста, взгляните на предоставленный мною ответ, может быть комбинация проблем, от конфигурации до использования неправильного экземпляра RestTemplate.   -  person ootero    schedule 21.12.2016


Ответы (4)


Для меня единственное, что мне нужно сделать, чтобы исправить проблему, - это добавить весеннюю аннотацию @LoadBalanced в

@Bean
@LoadBalanced
public RestTemplate restTemplate() {
    return new RestTemplate();
}
person Manu    schedule 22.05.2018

Как отмечалось ранее, я полагаю, что вам может не хватать Ribbon конфигурации в eureka-client-1.

Сначала я бы переехал:

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

классу конфигурации.

Добавьте конфигурацию Ribbon в application.yml, например:

the-eureka-client-2:
   ribbon:
     # Eureka vipAddress of the target service
     DeploymentContextBasedVipAddresses: eureka-client-2

     #listOfServers: localhost:${SERVER.PORT}
     NIWSServerListClassName: com.netflix.niws.loadbalancer.DiscoveryEnabledNIWSServerList

     # Interval to refresh the server list from the source (ms)
     ServerListRefreshInterval: 30000

Внедрить restTemplate в класс HystrixDemoService вместо создания нового экземпляра для каждого запроса. RestTemplate является потокобезопасным:

@Component
public class HystrixDemoService {   

    @Autowired
    public RestTemplate restTemplate;
...
    @HystrixCommand(fallbackMethod = "getFallbackCustomerName")
    public String getCustomerName() {
      URI uri = URI.create("http://eureka-client-2");
      return this.restTemplate.getForObject(uri, String.class);
    }

где the-eureka-client-2 - это ключ, который сопоставляется с зарегистрированной службой с именем: eureka-client-2

Я писал о Регистрация и обнаружение микросервисов с использованием Spring Cloud Eureka, Ribbon и Feign, который также включает исходный код сервера Discovery и двух клиентов, разработанных с использованием Jersey 1 и Spring MVC.

person ootero    schedule 21.12.2016

Изменения ниже сработали для меня.

@SpringBootApplication
public class DemoPranayEurekaClient1Application {
    public static void main(String[] args) {
        SpringApplication.run(DemoPranayEurekaClient1Application.class, args);
    }
}

@EnableDiscoveryClient
@EnableCircuitBreaker
@RestController
class HystrixDemoApplication {
    @Autowired
    HystrixDemoService hystrixDemoService;

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

    @RequestMapping("/")
    public String name() {
        String str = hystrixDemoService.getCustomerName();
        return "I'm A talking to "+str;
    }

}

Ниже приведена строка кода, которая используется для выбора экземпляра eureka-client-2 ...

Экземпляр ServiceInstance = loadBalancer.choose ("эврика-клиент-2");

@Component
public class HystrixDemoService {

    @Autowired
    private LoadBalancerClient loadBalancer;

    @HystrixCommand(fallbackMethod = "getFallbackCustomerName")
    public String getCustomerName() {
        RestTemplate restTemplate = new RestTemplate();
        ServiceInstance instance = loadBalancer.choose("eureka-client-2");
        URI uri = instance.getUri();
        return restTemplate.getForObject(uri, String.class);
    }

    public String getFallbackCustomerName() {
        System.out.println("coming inside fallback method");
        return "Resillient Customer";
    }
}
person prranay    schedule 24.12.2016

Решение в том, что вы упускаете @LoadBalanced.
если вы используете Spring Boot или Sprint Cloud, я предпочитаю использовать RestTemplateBuilder builder

  @Bean
  @LoadBalanced
  public RestTemplate restTemplate(RestTemplateBuilder builder) {
    return builder.build();
  }

Если вы используете только пружину, я предлагаю вам new RestTemplate()

  @Bean
  @LoadBalanced
  public RestTemplate restTemplate() {
      return new RestTemplate();
  }
person Zgpeace    schedule 04.03.2020