golang 注册和获取consul服务,api服务注册获取以及grpc注册获取

1.普通接口注册服务

执行后就会出现下面那个服务,他这里会检查这个服务是否可用,不可用就会自动剔除

package main

import (
    "fmt"
    "net/http"
    consulapi "github.com/hashicorp/consul/api"
)

const (
    consulAddress = "124.70.156.31:8500"
    localIP       = "124.70.156.31"
    localPort     = 3001
)

func consulRegister() {
    // 创建连接consul服务配置
    config := consulapi.DefaultConfig()
    config.Address = consulAddress
    client, err := consulapi.NewClient(config)
    if err != nil {
        fmt.Println("consul client error : ", err)
    }

    // 创建注册到consul的服务到
    registration := new(consulapi.AgentServiceRegistration)
    registration.ID = "shitingbao"
    registration.Name = "service_shitingbao"//根据这个名称来找这个服务
    registration.Port = localPort
    registration.Tags = []string{"shitingbao_test_service"}//这个就是一个标签,可以根据这个来找这个服务,相当于V1.1这种
    registration.Address = localIP

    // 增加consul健康检查回调函数
    check := new(consulapi.AgentServiceCheck)
    check.HTTP = fmt.Sprintf("http://%s:%d", registration.Address, registration.Port)
    check.Timeout = "5s"                         //超时
    check.Interval = "5s"                        //健康检查频率
    check.DeregisterCriticalServiceAfter = "30s" // 故障检查失败30s后 consul自动将注册服务删除
    registration.Check = check

    // 注册服务到consul
    err = client.Agent().ServiceRegister(registration)
}

//Handler 3001
func Handler(w http.ResponseWriter, r *http.Request) {
    w.Write([]byte("you are visiting health check api:3001"))
}

//ServerLoad 启动
func ServerLoad() {
    consulRegister()
    //定义一个http接口
    http.HandleFunc("/", Handler)
    err := http.ListenAndServe(":3001", nil)
    if err != nil {
        fmt.Println("error: ", err.Error())
    }
}

2.grpc注册入consul

grpc需要多做一步,因为consul需要一个健康检查,在api中验证是否可用是可以直接检查就行了,但是对于rpc的这种协议不一样处理,这里需要写一个自定义的检查函数,给consul检查。这个函数需要实现consul包中的RegisterHealthServer接口,grpc服务的代码就不展示了,这里展示了注册grpc的过程

package main

import (
    "context"
    "fmt"
    "log"
    "net"
    stb_server "stb_consul/external_service/stb_server"
    "stb_consul/external_service/stbserver"

    "github.com/hashicorp/consul/api"
    "github.com/sirupsen/logrus"
    "google.golang.org/grpc"
    "google.golang.org/grpc/health/grpc_health_v1"
)

// HealthImpl 健康检查实现
type HealthImpl struct{}

// Check 实现健康检查接口,这里直接返回健康状态,这里也可以有更复杂的健康检查策略,比如根据服务器负载来返回
func (h *HealthImpl) Check(ctx context.Context, req *grpc_health_v1.HealthCheckRequest) (*grpc_health_v1.HealthCheckResponse, error) {
    return &grpc_health_v1.HealthCheckResponse{
        Status: grpc_health_v1.HealthCheckResponse_SERVING,
    }, nil
}

//Watch 这个没用,只是为了让HealthImpl实现RegisterHealthServer内部的interface接口
func (h *HealthImpl) Watch(req *grpc_health_v1.HealthCheckRequest, w grpc_health_v1.Health_WatchServer) error {
    return nil
}

//grpc开启
func externalServer() {
    lis, err := net.Listen("tcp", ":3001")
    if err != nil {
        logrus.Info("外置服务开启失败:", err)
        panic(err)
    }
    logrus.WithFields(logrus.Fields{
        "tcp": ":3001",
    }).Info("external server")
    s := grpc.NewServer()
    stbserver.RegisterStbServerServer(s, &stb_server.StbServe{})
    grpc_health_v1.RegisterHealthServer(s, &HealthImpl{})//比普通的grpc开启多了这一步
    s.Serve(lis)
    log.Println("grpc start")
}

//grpc注册进consul
func grpcRegister() {
    config := api.DefaultConfig()
    config.Address = consulAddress
    client, err := api.NewClient(config)
    if err != nil {
        panic(err)
    }
    agent := client.Agent()

    reg := &api.AgentServiceRegistration{
        ID:      fmt.Sprintf("%v-%v-%v", "StbServe", localIP, localPort), // 服务节点的名称
        Name:    fmt.Sprintf("grpc.health.v1.%v", "StbServe"),            // 服务名称
        Tags:    []string{"StbServe"},                                    // tag,可以为空
        Port:    localPort,                                               // 服务端口
        Address: localIP,                                                 // 服务 IP
        Check: &api.AgentServiceCheck{ // 健康检查
            Interval: "5s", // 健康检查间隔
            // grpc 支持,执行健康检查的地址,service 会传到 Health.Check 函数中
            GRPC:                           fmt.Sprintf("%v:%v/%v", localIP, localPort, "StbServe"),
            DeregisterCriticalServiceAfter: "5s", // 注销时间,相当于过期时间
        },
    }

    if err := agent.ServiceRegister(reg); err != nil {
        panic(err)
    }
}

func grpcLoad() {
    grpcRegister()
    externalServer()
}

3.服务查看

你可以在你的consul的UI中看见这个服务

image
image

4.发现服务

发现服务中的代码都是一样的,如下

package main

import (
    "fmt"

    "github.com/hashicorp/consul/api"
    "github.com/sirupsen/logrus"
)

func client() {
    var lastIndex uint64
    config := api.DefaultConfig()
    config.Address = "124.70.156.31:8500" //consul server

    client, err := api.NewClient(config)
    if err != nil {
        fmt.Println("api new client is failed, err:", err)
        return
    }
    services, metainfo, err := client.Health().Service("service_shitingbao", "shitingbao_test_service", true, &api.QueryOptions{
        WaitIndex: lastIndex, // 同步点,这个调用将一直阻塞,直到有新的更新
    })
    if err != nil {
        logrus.Panic("error retrieving instances from Consul:", err)
    }
    lastIndex = metainfo.LastIndex

    for _, service := range services {
        fmt.Println("service.Service.Address:", service.Service.Address, "service.Service.Port:", service.Service.Port)
    }
}

这里会输出ServiceName为‘service_shitingbao’,tag标签为shitingbao_test_service的服务,这个service_shitingbao就是上面红框里面的那个服务名称,对应的tag标签要对应上,没有就是空,不然获取不到

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容