2020-11-06

多语言微服务版本管理实践

2020_11_06_OlGYFbXEtO.png

版本化、优雅、简单实现微服务之间的rpc通讯

首先划分业务架构

淘宝、京东、天猫属于业务层

订单、支付属于基础服务


2020_11_06_GrT8GQ84ww.png

实践

本文研究的是业务层(php)如何优雅的通过rpc(grpc, protobuf)和基础服务(go)通讯

现在有2个底层服务(商品、订单GOLANG),一个业务中台(laravel),

需要实现

  1. php和golang之间0配置调用,版本管理
  2. rpc和rpc之间proto版本化管理

goods 服务的proto,声明了一个 带有 getOneByTypeIphone svc

syntax="proto3";

package goods;

option go_package="github.com/DuC-cnZj/goods_proto";
option php_metadata_namespace = "Goods";

message Request {
  int32 type = 1;
}

message Response {
  string code = 1;
  string msg = 2;
}

service Iphone {
  rpc getOneByType(Request) returns (Response);
}

订单服务的proto,声明了一个带有 list create 的订单服务

syntax="proto3";

package order;

import "google/protobuf/empty.proto";

option go_package="github.com/DuC-cnZj/order_protos";
option php_metadata_namespace = "Order";

message Request {
  string num = 1;
  string user_name = 2;
}

message Response {
  int32 code = 1;
  string data = 2;
}

service Order {
  rpc list(google.protobuf.Empty) returns (Response);
  rpc create(Request) returns (Response);
}

生成对应的php版本的pb.go文件,那么php应该如何接入呢?

直接服务pb文件到php项目的目录下?不可行,如果有多个php项目,复用性为0,修改起来麻烦,无法版本化管理。

那么该怎么做呢?

答案是通过composer,生成 package,并且在 ci/cd 阶段自动化打包提交,这时候就会得到可复用的,版本化的composer包。

php client 使用麻烦怎么办?

每次使用都需要这样写,太麻烦了

$client = new IphoneClient("rpc:host", []);
[$data, $response] = $client->getOneByType($request)->wait();
if ($response->code == \Grpc\CALL_OK) {
  if ($asArray) {
    return json_decode($data->serializeToJsonString(), true);
  }

  return $data;
}

throw new \Exception("Goods rpc client error: " . $response->details, $response->code);

能否用一行搞定?类似 IphoneClient::getOneByType(['type' => 1]),或许你觉得不可能,但是实际上可以。

如果你使用laravel框架,肯定会惊叹它的Facades是多么的好用,它的服务自动发现是多么神奇,如果能用到grpc中那么该多好。

在这里我推荐一个pkg duc_cnzj/rpc-facades-generator,自动帮你生成grpc facades,实现零修改,自动生成如下代码

<?php

namespace Goods\Facades;

use Illuminate\Support\Facades\Facade;

/**
 *
 * @method static \Goods\Response|array getOneByType($data = [], bool $asArray = true)
 *
 * Class IphoneClient
 */
class IphoneClient extends Facade
{
    protected static function getFacadeAccessor()
    {
        return \Goods\Services\IphoneClientService::class;
    }
}

那么你可能会问,rpc调用的地址在哪里配?

<?php

namespace Goods;

use Goods\IphoneClient;


class ServiceProvider extends \Illuminate\Support\ServiceProvider
{
    public function register()
    {
        $this->app->singleton(IphoneClient::class);
        $this->app->when(IphoneClient::class)
            ->needs('$hostname')
            ->give(env("GOODS_HOST", ""));
        $this->app->when(IphoneClient::class)
            ->needs('$opts')
            ->give([
                'credentials' => \Grpc\ChannelCredentials::createInsecure(),
            ]);
    }
}

没错,你只需要在 .env 中配置 GOODS_HOST 就可以了,其他的不用管!pkg帮你自动生成!

搞定了php和go之间的版本化管理,那么我们再来谈谈go和go微服务之间的proto版本管理,和php相同原理,这里我们也使用pkd来管理proto文件通过ci/cd 生成pb文件,并且用git打包上传到一个专门的proto仓库,实现版本化管理。

源码

rpc-facades-generator grpc laravel facades 生成器,只需要 require --dev 即可

order_protos order 服务的proto文件仓库

micro-order order 服务

goods_proto goods 服务的proto仓库

micro-goods goods服务

核心实现要点: 通过 ci/cd 生成并且上传对应语言的 pb 文件,并且通过pkg管理起来

优点

  1. 版本化
  2. 可复用
  3. 简单

思考?

单体golang微服务仓库对比多仓库

单体仓库cicd麻烦,并且团队合作相比多仓库难

单体仓库构建流程麻烦不推荐

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