基于Go/Grpc/kubernetes/Istio开发微服务的最佳实践尝试 - 2/3

基于Go/Grpc/kubernetes/Istio开发微服务的最佳实践尝试 - 1/3

基于Go/Grpc/kubernetes/Istio开发微服务的最佳实践尝试 - 2/3

基于Go/Grpc/kubernetes/Istio开发微服务的最佳实践尝试 - 3/3

项目地址:https://github.com/janrs-io/Jgrpc


转载请注明来源:https://janrs.com/ugj7


在上一部分中,我们创建了微服务的目录结构并实现了微服务pongservice

这部分我们继续实现一个名为pingservice的微服务,访问上一节已经部署好的pongservice微服务。

创建一个新的微服务非常简单,只需复制之前创建的pongservice微服务,然后做一些小改动。

项目结构

这部分最终的目录结构如下:

pingservice
├── buf.gen.yaml
├── cmd
│   ├── main.go
│   └── server
│       ├── grpc.go
│       ├── http.go
│       ├── run.go
│       ├── wire.go
│       └── wire_gen.go
├── config
│   ├── client.go
│   ├── config.go
│   └── config.yaml
├── genproto
│   └── v1
│       ├── gw
│       │   └── pingservice.pb.gw.go
│       ├── pingservice.pb.go
│       └── pingservice_grpc.pb.go
├── go.mod
├── go.sum
├── proto
│   ├── buf.lock
│   ├── buf.yaml
│   └── v1
│       ├── pingservice.proto
│       └── pingservice.yaml
└── service
    ├── client.go
    └── server.go

9 directories, 21 files

开始

复制

src 目录下执行如下复制命令:

cp -R pongservice pingservice

删除 wire_gen.go

删除 pingservice/cmd/server 目录中的 wire_gen.go 文件。

修改 go.mod module

修改go.mod文件的模块,如下代码所示:

module github.com/janrs-io/Jgrpc/src/pingservice

生成 proto

删除pingservice/proto/v1目录下的pongservice.protopongservice.yaml文件以及整个genproto文件夹。

然后重新创建pingservice.protopingservice.yaml文件,代码如下:

pingservice.proto code:

syntax = "proto3";

package proto.v1;

option go_package = "github.com/janrs-io/Jgrpc/src/pingservice/genproto/v1";

service PingService {
  rpc Ping(PingRequest) returns(PingResponse){}
}

message PingRequest {
  string msg = 1 ;
}

message PingResponse {
  string msg = 1;
}

pingservice.yaml code:

type: google.api.Service
config_version: 3

http:
  rules:
    - selector: proto.v1.PingService.Ping
      get: /ping.v1.ping

修改pingservice 目录下的 buf.gen.yaml 文件。 修改后的完整代码如下:

version: v1
plugins:
  - plugin: go
    out: genproto/v1
    opt:
      - paths=source_relative
  - plugin: go-grpc
    out: genproto/v1
    opt:
      - paths=source_relative
  - plugin: grpc-gateway
    out: genproto/v1/gw
    opt:
      - paths=source_relative
      - grpc_api_configuration=proto/v1/pingservice.yaml
      - standalone=true

pingservice目录下执行以下命令生成 proto 文件:

buf generate proto/v1

执行命令后,将重新生成genproto目录,并自动创建*pb.go文件。

修改 import 路径和所有代码

检查所有文件的导入,将导入路径的pongservice修改为pingservice

将所有代码的Pong/pong改为Ping/ping,直到没有错误为止。

修改 config.yaml

修改 config 目录下的config.yaml文件。
grpc 的服务器端口更改为 50052,将 http 的服务器端口更改为 9002
grpc 的服务名改为ping-grpc,将 http 的服务名改为ping-http

修改后的完整代码如下:

# grpc config
grpc:
  host: ""
  port: ":50052"
  name: "ping-grpc"

# http config
http:
  host: ""
  port: ":9002"
  name: "ping-http"

生成 generate inject

pingservice 目录中执行以下 wire 命令以重新生成依赖注入文件:

wire ./...

引入 pongservier 服务的 go.mod

我们要在 pingservice 这个微服务中访问 pongservicegrpc 服务,所以需要导入 pongservicego.mod

修改pingservice目录下的go.mod文件,添加导入pongservice的代码,如下:

module github.com/janrs-io/Jgrpc/src/pingservice

go 1.19

replace (
    pongservice => ../pongservice
)

require (
    pongservice v0.0.0
    github.com/google/wire v0.5.0
    github.com/grpc-ecosystem/grpc-gateway/v2 v2.15.2
    github.com/spf13/viper v1.15.0
    google.golang.org/grpc v1.54.0
    google.golang.org/protobuf v1.30.0
)

// the other required package...

修改 config.yaml

修改pingservice/config目录下的config.yaml文件,添加如下代码:

# service client
client:
  pong: ":50051"

修改后的完整代码如下:

# grpc config
grpc:
  host: ""
  port: ":50052"
  name: "ping-grpc"

# http config
http:
  host: ""
  port: ":9002"
  name: "ping-http"

# service client
client:
  pong: ":50051"

生成 client.go

pingservice/config目录下生成client.go文件,添加如下代码:

package config

// Client Client service config
type Client struct {
    Pong string `json:"pong" yaml:"pong"`
}

修改 config.go

Config 结构体中添加一个 Client 字段,代码如下:

Client Client `json:"client" yaml:"client"`

修改后的完整代码如下:

package config

import (
    "net/http"

    "github.com/spf13/viper"
    "google.golang.org/grpc"
)

// Config Service config
type Config struct {
    Grpc   Grpc   `json:"grpc" yaml:"grpc"`
    Http   Http   `json:"http" yaml:"http"`
    Client Client `json:"client" yaml:"client"`
}

// NewConfig Initial service's config
func NewConfig(cfg string) *Config {

    if cfg == "" {
        panic("load config file failed.config file can not be empty.")
    }

    viper.SetConfigFile(cfg)

    // Read config file
    if err := viper.ReadInConfig(); err != nil {
        panic("read config failed.[ERROR]=>" + err.Error())
    }
    conf := &Config{}
    // Assign the overloaded configuration to the global
    if err := viper.Unmarshal(conf); err != nil {
        panic("assign config failed.[ERROR]=>" + err.Error())
    }

    return conf

}

// Grpc Grpc server config
type Grpc struct {
    Host   string `json:"host" yaml:"host"`
    Port   string `json:"port" yaml:"port"`
    Name   string `json:"name" yaml:"name"`
    Server *grpc.Server
}

// Http Http server config
type Http struct {
    Host   string `json:"host" yaml:"host"`
    Port   string `json:"port" yaml:"port"`
    Name   string `json:"name" yaml:"name"`
    Server *http.Server
}

修改 client.go

修改pingservice/service目录下的client.go文件,添加如下代码:

// NewPongClient New pong service client
func NewPongClient(conf *config.Config) (pongclientv1.PongServiceClient, error) {

    ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
    defer cancel()

    conn, err := grpc.DialContext(ctx, conf.Client.Pong, grpc.WithTransportCredentials(insecure.NewCredentials()))
    if err != nil {
        fmt.Println("dial auth server failed.[ERROR]=>" + err.Error())
        return nil, err
    }
    client := pongclientv1.NewPongServiceClient(conn)
    return client, nil

}

修改后的完整代码如下:

package service

import (
    "context"
    "fmt"
    "time"

    "google.golang.org/grpc"
    "google.golang.org/grpc/credentials/insecure"

    "github.com/janrs-io/Jgrpc/src/pingservice/config"
    v1 "github.com/janrs-io/Jgrpc/src/pingservice/genproto/v1"
    pongclientv1 "github.com/janrs-io/Jgrpc/src/pongservice/genproto/v1"
)

// NewClient New service's client
func NewClient(conf *config.Config) (v1.PingServiceClient, error) {

    serverAddress := conf.Grpc.Host + conf.Grpc.Port
    ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
    defer cancel()

    conn, err := grpc.DialContext(ctx, serverAddress, grpc.WithTransportCredentials(insecure.NewCredentials()))
    if err != nil {
        return nil, err
    }
    client := v1.NewPingServiceClient(conn)
    return client, nil

}

// NewPongClient New pong service client
func NewPongClient(conf *config.Config) (pongclientv1.PongServiceClient, error) {

    ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
    defer cancel()

    conn, err := grpc.DialContext(ctx, conf.Client.Pong, grpc.WithTransportCredentials(insecure.NewCredentials()))
    if err != nil {
        fmt.Println("dial auth server failed.[ERROR]=>" + err.Error())
        return nil, err
    }
    client := pongclientv1.NewPongServiceClient(conn)
    return client, nil

}

修改 server.go

修改pingservice/service目录下的server.go文件,修改后的完整代码如下:

package service

import (
    "context"
    "google.golang.org/grpc/grpclog"

    "github.com/janrs-io/Jgrpc/src/pingservice/config"
    v1 "github.com/janrs-io/Jgrpc/src/pingservice/genproto/v1"
    pongclientv1 "github.com/janrs-io/Jgrpc/src/pongservice/genproto/v1"
)

// Server Server struct
type Server struct {
    v1.UnimplementedPingServiceServer
    pingClient v1.PingServiceClient
    pongClient pongclientv1.PongServiceClient
    conf       *config.Config
}

// NewServer New service grpc server
func NewServer(
    conf *config.Config,
    pingClient v1.PingServiceClient,
    pongClient pongclientv1.PongServiceClient,
) v1.PingServiceServer {
    return &Server{
        pingClient: pingClient,
        pongClient: pongClient,
        conf:       conf,
    }
}

func (s *Server) Ping(ctx context.Context, req *v1.PingRequest) (*v1.PingResponse, error) {
    pongReq := &pongclientv1.PongRequest{Msg: "request from ping service"}
    pongResp, err := s.pongClient.Pong(ctx, pongReq)
    if err != nil {
        grpclog.Error("connect pong failed.[ERROR]=>" + err.Error())
        return nil, err
    }

    return &v1.PingResponse{
        Msg: "response ping msg:" + req.Msg + " and msg from pong service is: " + pongResp.Msg,
    }, nil
}

修改 wire.go

修改pingservice/cmd/serverwire.go文件,添加service.NewPongClient依赖注入。 代码如下:

service.NewPongClient

修改后的完整代码如下:

//go:build wireinject
// +build wireinject

package server

import (
    "github.com/google/wire"

    "github.com/janrs-io/Jgrpc/src/pingservice/config"
    v1 "github.com/janrs-io/Jgrpc/src/pingservice/genproto/v1"
    "github.com/janrs-io/Jgrpc/src/pingservice/service"
)

// InitServer Inject service's component
func InitServer(conf *config.Config) (v1.PingServiceServer, error) {

    wire.Build(
        service.NewPongClient,
        service.NewClient,
        service.NewServer,
    )

    return &service.Server{}, nil

}

pingservice 目录下执行以下 wire 命令重新生成依赖注入文件:

如果出现 go.mod 引入错误,只需在 pingservice 目录中再次运行 go mod tidy

wire ./...

启动 service

分别在pongservice目录和pingservice目录下执行go run命令。

go run cmd/main.go

在浏览器中输入以下请求地址:

127.0.01:9002/ping.v1.ping?msg=best practice

一切正确的情况下返回以下 json 数据:

{
    "msg": "response ping msg:best practice and msg from pong service is: response pong msg:request from ping service"
}

总结

这部分我们新建一个 pingservice 微服务,实现访问 pongservicegrpc 服务。

相信通过这两次创建微服务的简单尝试,你一定觉得基于GoGrpc开发微服务并不难。

在下一部分中,我们将利用 Jenkins/Gitlab/HarborKubernets/Istio 进行 devopsCICD 部署。


转载请注明来源:https://janrs.com/ugj7

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

推荐阅读更多精彩内容