Files
common/consul/consul.go

168 lines
4.0 KiB
Go
Raw Normal View History

2025-11-25 11:51:16 +08:00
package consul
import (
"context"
2025-11-28 08:52:22 +08:00
"errors"
"fmt"
"net"
2025-12-09 17:55:08 +08:00
"sync"
2026-04-01 11:37:43 +08:00
"time"
2025-11-25 11:51:16 +08:00
"github.com/gogf/gf/contrib/registry/consul/v2"
"github.com/gogf/gf/v2/frame/g"
"github.com/gogf/gf/v2/net/gsel"
2025-11-25 11:51:16 +08:00
"github.com/gogf/gf/v2/net/gsvc"
2025-11-28 08:52:22 +08:00
"github.com/gogf/gf/v2/util/grand"
2025-11-25 11:51:16 +08:00
)
2026-04-01 11:37:43 +08:00
var (
registry gsvc.Registry
consulAddr string
reconnectMutex sync.RWMutex
reconnectDone chan struct{}
)
2025-12-09 17:55:08 +08:00
2026-04-01 11:37:43 +08:00
// connectConsul 连接 Consul
func connectConsul(ctx context.Context) error {
reconnectMutex.Lock()
defer reconnectMutex.Unlock()
var err error
registry, err = consul.New(consul.WithAddress(consulAddr))
if err != nil {
g.Log().Errorf(ctx, "❌ Consul 连接失败: %v", err)
return err
}
gsvc.SetRegistry(registry)
gsel.SetBuilder(gsel.NewBuilderRoundRobin())
g.Log().Infof(ctx, "✅ Consul 初始化成功: %s", consulAddr)
// 启动健康检查和自动重连
go startHealthCheckAndReconnect()
return nil
}
// startHealthCheckAndReconnect 启动健康检查和自动重连
func startHealthCheckAndReconnect() {
if reconnectDone != nil {
close(reconnectDone)
}
reconnectDone = make(chan struct{})
ticker := time.NewTicker(5 * time.Second)
defer ticker.Stop()
ctx := context.Background()
for {
select {
case <-ticker.C:
// 检查服务发现是否正常工作
if checkConsulHealth(ctx) {
continue
}
g.Log().Warning(ctx, "⚠️ Consul 连接异常,尝试重新连接...")
if err := connectConsul(ctx); err != nil {
g.Log().Errorf(ctx, "❌ Consul 重连失败: %v,5秒后重试...", err)
}
case <-reconnectDone:
g.Log().Info(ctx, "🛑 Consul 健康检查已停止")
2025-12-09 17:55:08 +08:00
return
}
2026-04-01 11:37:43 +08:00
}
}
// checkConsulHealth 检查 Consul 健康状态
func checkConsulHealth(ctx context.Context) bool {
reconnectMutex.RLock()
defer reconnectMutex.RUnlock()
if registry == nil {
return false
}
// 尝试获取服务列表来检测连接是否正常
services, err := registry.Search(ctx, gsvc.SearchInput{})
if err != nil {
g.Log().Debugf(ctx, "Consul 健康检查失败: %v", err)
return false
}
g.Log().Debugf(ctx, "✅ Consul 健康检查通过,发现 %d 个服务", len(services))
return true
2025-12-09 17:55:08 +08:00
}
2025-11-25 11:51:16 +08:00
func init() {
2026-04-01 11:37:43 +08:00
consulAddr = g.Cfg().MustGet(context.Background(), "consul.address").String()
if consulAddr == "" {
g.Log().Warning(context.Background(), "⚠️ Consul 配置未找到,跳过初始化")
return
}
if err := connectConsul(context.Background()); err != nil {
g.Log().Errorf(context.Background(), "❌ Consul 初始化失败: %v", err)
}
2025-11-25 11:51:16 +08:00
}
func getLocalIP() (string, error) {
// 获取本机所有网络接口
addrs, err := net.InterfaceAddrs()
if err != nil {
return "", err
}
for _, addr := range addrs {
// 检查是否是IP地址
if ipNet, ok := addr.(*net.IPNet); ok && !ipNet.IP.IsLoopback() {
if ipNet.IP.To4() != nil {
// 返回第一个非回环的IPv4地址
return ipNet.IP.String(), nil
}
}
}
return "", fmt.Errorf("无法找到本地IP地址")
}
2025-12-03 14:34:27 +08:00
func getInstanceAddrByIp(ctx context.Context, ip string, services []gsvc.Service) (addr string) {
for _, s := range services {
if s.GetEndpoints()[0].Host() == ip {
2025-12-03 14:34:27 +08:00
addr = s.GetEndpoints()[0].String()
return
}
}
return
}
2025-11-28 08:52:22 +08:00
func GetInstanceAddr(ctx context.Context, name string) (addr string, err error) {
watch, err := gsvc.GetRegistry().Watch(ctx, name)
2026-04-01 11:37:43 +08:00
if err != nil {
err = errors.New("获取服务监听器失败")
return
}
2025-11-28 08:52:22 +08:00
service, err := watch.Proceed()
if err != nil || service == nil {
2026-04-01 11:37:43 +08:00
err = errors.New("获取服务实例失败")
2025-11-28 08:52:22 +08:00
return
}
2025-12-03 14:34:27 +08:00
//优先使用客户端IP获取实例(前后端在同一台机器调试)
2025-12-03 14:40:03 +08:00
addr = getInstanceAddrByIp(ctx, g.RequestFromCtx(ctx).GetClientIp(), service)
2025-12-03 14:34:27 +08:00
if !g.IsEmpty(addr) {
return
}
//优先使用gateway同IP的服务实例(前后端不同机器调试)
addr, err = getLocalIP()
if err != nil {
return
}
2025-12-03 14:34:27 +08:00
addr = getInstanceAddrByIp(ctx, addr, service)
if !g.IsEmpty(addr) {
return
}
2025-12-03 14:34:27 +08:00
//随机获取一个服务实例
maxService := grand.N(0, len(service)-1)
addr = service[maxService].GetEndpoints()[0].String()
2025-11-28 08:52:22 +08:00
return
}