Ocelot 发现服务总是失败的解决办法

一、问题 

今天用 Ocelot + Consul 项目,进行微服务实践,可是 Ocelot 的发现服务总是失败。

 

二、分析问题

2.1、分析方法:

      不得不下载了 Ocelot 源码进行追踪排查。

 

2.2、源码分析

源码对应文件为 Ocelot-develop\src\Ocelot.Provider.Consul\Consul.cs

public async Task> Get()
        {
            var queryResult = await _consul.Health.Service(_config.KeyOfServiceInConsul, string.Empty, true);

            var services = new List();

            foreach (var serviceEntry in queryResult.Response)
            {
                if (IsValid(serviceEntry))
                {
                    var nodes = await _consul.Catalog.Nodes();
                    if (nodes.Response == null)
                    {
                        services.Add(BuildService(serviceEntry, null));
                    }
                    else
                    {
                        var serviceNode = nodes.Response.FirstOrDefault(n => n.Address == serviceEntry.Service.Address);
                        services.Add(BuildService(serviceEntry, serviceNode));
                    }
                }
                else
                {
                    _logger.LogWarning($"Unable to use service Address: {serviceEntry.Service.Address} and Port: {serviceEntry.Service.Port} as it is invalid. Address must contain host only e.g. localhost and port must be greater than 0");
                }
            }

            return services.ToList();
        }

        private Service BuildService(ServiceEntry serviceEntry, Node serviceNode)
        {
            return new Service(
                serviceEntry.Service.Service,
                new ServiceHostAndPort(serviceNode == null ? serviceEntry.Service.Address : serviceNode.Name, serviceEntry.Service.Port),
                serviceEntry.Service.ID,
                GetVersionFromStrings(serviceEntry.Service.Tags),
                serviceEntry.Service.Tags ?? Enumerable.Empty());
        }

  

第3行,获得已注测的健康的服务;第11行,获得已注册 Consul 节点的电脑。

第18行,如果当前服务与Consul节点是在同一台电脑上,则返回电脑节点 ServiceNode

第31行中的 BuildSevrice 函数完成最后的 service信息创建, 

第35行  如果找到了 ServiceNode,则返回对应的电脑名字 hostname,  如果没找到则返回对应的 ipAddress.

就是说:服务注和Consul 在同一台电脑上则返回hostname , 在不同电脑上则返回服务所在电脑的 ipAddress. 

 Ocelot 发现服务总是失败的解决办法_第1张图片

 

2.3 问题根源:

是 Consul 注册时,hostname 参数由 -node参数指定。

当服务和 Consul 在同一台电脑上时,Ocelot 最终变换成  http://hostname:port/url  的形式进行访问。

 

三、解决问题

3.1、Consul 的配置参数注意事项

注意而 hostname 是由Consul 的 -node 参数指定的!

所以,

consul agent -server -datacenter=dc1 -bootstrap  -data-dir ./data  -ui  -node=n1 -bind 192.168.11.211 -client=0.0.0.0

  其中 -node=n1 是一个大坑。应该略去,系统会自己设置为自己的主机名字 hostname。

所以我实际上用了配置文件 node1.json,也是去掉了该项。

{
  "datacenter": "dc1",
  "data_dir": "c:/data/app/consul/node1",
  "log_level": "INFO",
  "server": true,
  "ui": true,
  "bind_addr": "192.168.11.211",
  "client_addr": "127.0.0.1",
  "advertise_addr": "192.168.11.211",
  "bootstrap_expect": 1,
  "ports":{
    "http": 8500,
    "dns": 8600,
    "server": 8300,
    "serf_lan": 8301,
    "serf_wan": 8302
    }
}

然后调用方式:

consul agent -config-dir=e:/consul/node1.json

然后用另一台服务器加入 

consul agent -data-dir /tmp/consul  -bind=192.168.11.246 -join 192.168.11.248

 

这时 Consul 的 web 管理界面为:

Ocelot 发现服务总是失败的解决办法_第2张图片

 

它会自动带上主机名: HNSever 和 LGB-PC

在这两台电脑上注册服务后,最终会变成  http://hostname:port/ + url 模板 的形式访问。

 

3.2、hostname 不能访问问题修改

如果 http://hostname:port/url,还是返回   错误代码 HTTP ERROR 500  访问失败。就是  hostname 不能转换为 ipaddress.

所以需要修改 windows 的 hosts 文件

打开系统目录:c:/windows/system32/drivers/etc找到hosts文件,打开hosts文件并在最后面添加一条记录

例如:

192.168.11.248  HNServer

192.168.11.211  LGB-PC

然后就能正常的发现服务了!

 

3.3 附上 Ocelot 的配置文件  ocelot.json 

{
  "ReRoutes": [
    {
      "DownstreamPathTemplate": "/api/{url}",
      "DownstreamScheme": "http",
      "ServiceName": "ProductService",
      "UseServiceDiscovery": true,
      "LoadBalancerOptions": {
        "Type": "LeastConnection"
      },
      "UpstreamPathTemplate": "/test/{url}",
      "UpstreamHttpMethod": [ "Post", "Get" ]
    }
  ],

  "GlobalConfiguration": {
    "ServiceDiscoveryProvider": {
      "Host": "localhost",
      "Port": 8500
    }
  }
}

 

 

四、参考

Ocelot + Consul实践

.net core Ocelot Consul 实现API网关 服务注册 服务发现 负载均衡

netcore ocelot api网关结合consul服务发现

Windows下主机名和IP映射设置

 

你可能感兴趣的:(Ocelot 发现服务总是失败的解决办法)