您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > Go语言

基于Go-Kit的Golang整洁架构实践

时间:2023-12-25 13:46:14  来源:微信公众号  作者:DeepNoMind

简介

Go是整洁架构(Clean Architecture)的完美选择。整洁架构本身只是一种方法,并没有告诉我们如何构建源代码,在尝试用新语言实现时,认识到这点非常重要。

自从我有了使用Ruby on RAIls的经验后,尝试了好几次编写第一个服务,而且我读过的大多数关于Go的整洁架构的文章都以一种非Go惯用的方式介绍结构布局。部分原因是这些例子中的包是根据层命名的——controller、model、service等等……如果你有这些类型的包,这是第一个危险信号,告诉你应用程序需要重新设计。在Go中,包名[2]应该描述包提供了什么,而不是包含了什么。

基于Go-Kit的Golang整洁架构实践

然后我开始了解go-kit,特别是它提供的发货示例[3],并决定在应用程序中实现相同的结构。后来,当我深入研究整洁架构(Clean Architecture)时,惊喜的发现go-kit方法是多么完美。

本文将介绍使用Go-Kit方法编写服务是如何符合整洁架构理念的。

整洁架构(Clean Architecture)

整洁架构(Clean Architecture)是由Bob大叔(Robert Martin)创建的一种软件架构设计。目标是分离关注点[4],允许开发人员封装业务逻辑,并使其独立于交付和框架机制。许多架构范例(如Onion和Hexagon架构)也有相同的目标,都是通过将软件划分成层来实现解耦。

基于Go-Kit的Golang整洁架构实践

圆圈中的箭头表示依赖规则。如果在外部循环中声明了某些内容,则不得在内部循环代码中引用。它既适用于实际的源代码依赖关系,也适用于命名。内层不依赖于任何外层。

外层包含低级组件,如UI、DB、传输或任何第三方服务,都可以被认为是应用程序的细节或插件。其思想是,外层的变化一定不会引起内层的任何变化。

不同模块/组件之间的依赖关系可以描述如下:

基于Go-Kit的Golang整洁架构实践

请注意,跨越边界的箭头只指向一个方向,边界后面的组件属于外层,包括controller、presenter和database。Interactor是实现BL的地方,可以将其视为用例层。

请注意Request Model和Response Model。这些对象分别描述了内层需要和返回的数据。controller将请求(在web的情况下是HTTP请求)转换为请求模型(Request Model),presenter将响应模型(Response Model)格式化为可以由视图模型(View Model)呈现的数据。

还要注意接口,用于反转控制流以与依赖规则相对应。Interactor通过Boundary接口与presenter对话,并通过Entity Gateway接口与数据层对话。

这是整洁架构的主要思想,通过依赖注入分离不同的层,使用依赖反转反转控制流。Interactor(BL)和实体对传输和数据层一无所知。这一点很重要,因为如果我们改变了外层细节,内层就不会发生级联变化。

什么是Go-Kit?

Go kit[5]是包的集合,可以帮助我们构建健壮、可靠、可维护的微服务。

对于来自Ruby on Rails的我来说,重要的是Go-Kit不是MVC框架。相反,它将应用程序分为三层:

  • Transport(传输)
  • Endpoint(端点)
  • Service(服务)

1.Transport

传输层是唯一熟悉交付机制(HTTP、gRPC、CLI…)的组件,这一点非常强大,因为我们可以通过提供不同的传输层来同时支持HTTP和CLI。

稍后我们将看到传输层是如何对应于上图中的controller和presenter的。

2.Endpoint

type Endpoint func(ctx context.Context, request interface{}) (response interface{}, err error)

端点层表示应用程序中的单个RPC,将交付连接到BL。这是根据输入和输出实际定义用例的地方,在整洁架构术语中是Request Model和Response Model。

 

注意,端点是接收请求并返回响应的函数,都是interface{},是RequestModel和ResponseModel。理论上也可以用类型参数(泛型)来实现。

 

3.Service

服务层(interactor)是实现BL的地方。服务层不知道端点层,服务层和端点层都不知道传输域(比如HTTP)。

Go-Kit提供了创建服务器(HTTP服务器/gRPC服务器等)的功能。例如HTTP:

package http // under go-kit/kit/transport/http

type DecodeRequestFunc func(context.Context, *http.Request) (request interface{}, err error)
type EncodeResponseFunc func(context.Context, http.ResponseWriter, interface{}) error

func NewServer(
  e endpoint.Endpoint,
  dec DecodeRequestFunc,
  enc EncodeResponseFunc,
  options ...ServerOption,
) *Server
  • DecodeRequestFunc将HTTP请求转换为Request Model,并且
  • EncodeResponseFunc格式化Response Model并将其编码到HTTP响应中。
  • 返回的*server实现http.Server(有ServeHTTP方法)。

传输层使用这个函数来创建http.Server,解码器和编码器在传输中定义,端点在运行时初始化。

简短示例:(基于发货示例[6])

简易服务

我们将描述一个具有两个API的简单服务,用于从数据层创建和读取文章,传输层是HTTP,数据层只是一个内存映射。可以在这里找到Github源代码[7]。

注意文件结构:

- inmem
  - articlerepo.go
- publishing
  - transport.go 
  - endpoint.go
  - service.go
  - formatter.go
- article
  - article.go

我们看看如何表示整洁架构的不同层。

  • article —— 这是实体层,不包含BL、数据层或传输层的知识。
  • inmem —— 这是数据层。
  • transport —— 这是传输层。
  • endpoint+service —— 组成了边界+交互器。

从服务开始:

import (
  "context"
  "fmt"
  "math/rand"
 
  "github.com/OrenRosen/gokit-example/article"
)

type ArticlesRepository interface {
   GetArticle(ctx context.Context, id string) (article.Article, error)
   InsertArticle(ctx context.Context, thing article.Article) error
}

type service struct {
   repo ArticlesRepository
}

func NewService(repo ArticlesRepository) *service {
   return &service{
      repo: repo,
   }
}

func (s *service) GetArticle(ctx context.Context, id string) (article.Article, error) {
   return s.repo.GetArticle(ctx, id)
}

func (s *service) CreateArticle(ctx context.Context, artcle article.Article) (id string, err error) {
   artcle.ID = generateID()
   if err := s.repo.InsertArticle(ctx, artcle); err != nil {
      return "", fmt.Errorf("publishing.CreateArticle: %w", err)
   }
   
   return artcle.ID, nil
}

func generateID() string {
  // code emitted
}

服务对交付和数据层一无所知,它不从外层(HTTP、inmem…)导入任何东西。BL就在这里,你可能会说这里没有真正的BL,这里的服务可能是冗余的,但需要记住这只是一个简单示例。

实体

package article

type Article struct {
   ID    string
   Title string
   Text  string
}

实体只是一个DTO,如果有业务策略或行为,可以添加到这里。

端点

endpoint.go定义了服务接口:

type Service interface {
   GetArticle(ctx context.Context, id string) (article.Article, error)
   CreateArticle(ctx context.Context, thing article.Article) (id string, err error)
}

然后为每个用例(RPC)定义一个端点。例如,对于获取文章::

type GetArticleRequestModel struct {
   ID string
}

type GetArticleResponseModel struct {
   Article article.Article
}

func MakeEndpointGetArticle(s Service) endpoint.Endpoint {
   return func(ctx context.Context, request interface{}) (response interface{}, err error) {
      req, ok := request.(GetArticleRequestModel)
      if !ok {
         return nil, fmt.Errorf("MakeEndpointGetArticle failed cast request")
      }
      
      a, err := s.GetArticle(ctx, req.ID)
      if err != nil {
         return nil, fmt.Errorf("MakeEndpointGetArticle: %w", err)
      }
      
      return GetArticleResponseModel{
         Article: a,
      }, nil
   }
}

注意如何定义RequestModel和ResponseModel,这是RPC的输入/输出。其思想是,可以看到所需数据(输入)和返回数据(输出),甚至无需读取端点本身的实现,因此我认为端点代表单个RPC。服务具有实际触发BL的方法,但是端点是RPC的应用定义。理论上,一个端点可以触发多个BL方法。

传输

transport.go注册HTTP路由:

type Router interface {
   Handle(method, path string, handler http.Handler)
}

func RegisterRoutes(router *httprouter.Router, s Service) {
   getArticleHandler := kithttp.NewServer(
      MakeEndpointGetArticle(s),
      decodeGetArticleRequest,
      encodeGetArticleResponse,
   )
   
   createArticleHandler := kithttp.NewServer(
      MakeEndpointCreateArticle(s),
      decodeCreateArticleRequest,
      encodeCreateArticleResponse,
   )
   
   router.Handler(http.MethodGet, "/articles/:id", getArticleHandler)
   router.Handler(http.MethodPost, "/articles", createArticleHandler)
}

传输层通过MakeEndpoint函数在运行时创建端点,并提供用于反序列化请求的解码器和用于格式化和编码响应的编码器。

例如:

func decodeGetArticleRequest(ctx context.Context, r *http.Request) (request interface{}, err error) {
   params := httprouter.ParamsFromContext(ctx)
   return GetArticleRequestModel{
      ID: params.ByName("id"),
   }, nil
}

func encodeGetArticleResponse(ctx context.Context, w http.ResponseWriter, response interface{}) error {
   res, ok := response.(GetArticleResponseModel)
   if !ok {
      return fmt.Errorf("encodeGetArticleResponse failed cast response")
   }
   
   formatted := formatGetArticleResponse(res)
   w.Header().Set("Content-Type", "Application/json")
   return json.NewEncoder(w).Encode(formatted)
}

func formatGetArticleResponse(res GetArticleResponseModel) map[string]interface{} {
  return map[string]interface{}{
    "data": map[string]interface{}{
      "article": map[string]interface{}{
        "id":    res.Article.ID,
        "title": res.Article.Title,
        "text":  res.Article.Text,
      },
    },
  }
}

你可能会问,为什么要使用另一个函数来格式化article,而不是在article实体上添加JSON标记?

这是个非常重要的问题。在article实体上添加JSON标记意味着article知道它是如何格式化的。虽然没有显式导入到HTTP,但打破了抽象,使实体包依赖于传输层。

例如,假设你想将对客户端的响应从"title"更改为"header",此更改仅涉及传输层。但是,如果此需求导致需要更改实体,则意味着该实体依赖于传输层,这就破坏了简洁架构原则。

我们看看这个简单应用的依赖关系图:

基于Go-Kit的Golang整洁架构实践

哇,你一定注意到了它们的相似性!article实体没有依赖关系(只有向内箭头)。外层,transport和inmem,只有指向BL和实体内层的箭头。

一切都和转换有关

跨界就是不同层次语言之间的转换。

BL层只使用应用语言,也就是说,只知道实体(没有HTTP请求或SQL查询)。为了跨越边界,流中的某个组件必须将应用语言转换为外层语言。

在传输层,有解码器(将HTTP请求转换为RequestModel的应用语言)和编码器(将应用语言ResponseModel转换为HTTP响应)。

数据层实现了repo,在我们的例子中是inmem。在另一种情况下,我们可能会让sql包负责将应用语言转换为SQL语言(查询和原始结果)。

"ing"包

你可能会说传输和服务不应该在同一个包中,因为它们位于不同的层,这是一个正确的论点。我从go-kit的shipping例子中取了一个例子,含有这种设计,ing包包含了传输/端点/服务,我发现从长远来看非常方便。话虽如此,如果我现在写的话,可能会用不同的包。

最后关于"尖叫架构(Screaming Architecture)"的一句话

Go非常适合简洁架构的另一个原因是包的命名及其思想。尖叫架构(Screaming Architecture) 和构建应用程序有关,以便应用程序的意图显而易见。在Ruby On Rails中,当查看结构时,就知道它是用Ruby On Rails框架编写的(控制器、模型、视图……)。在我们的应用程序中,当查看结构时,可以看出这是一个关于文章的应用程序,有发布用例,并使用inmem数据层。

总结

简洁架构只是一种方法,并不会告诉你如何构建源代码,其实现艺术在于了解所用语言的使用惯例和工具。希望这篇文章对你有所帮助,重要的是要意识到,那些争论设计问题解决方案的文章并不总是对的,当然也包括这篇



Tags:Golang   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系,我们将及时更正、删除。
▌相关推荐
慢聊Golang协程池Ants实现原理
大家都知道goroutine 是 Go语言中的轻量级线程实现,由 Go 运行时(runtime)管理,Go 程序会智能地将 goroutine 中的任务合理地分配给每个 CPU。创建一个goroutine大小大概在2k左...【详细内容】
2023-12-27  Search: Golang  点击:(94)  评论:(0)  加入收藏
基于Go-Kit的Golang整洁架构实践
简介Go是整洁架构(Clean Architecture)的完美选择。整洁架构本身只是一种方法,并没有告诉我们如何构建源代码,在尝试用新语言实现时,认识到这点非常重要。自从我有了使用Ruby o...【详细内容】
2023-12-25  Search: Golang  点击:(154)  评论:(0)  加入收藏
Golang清晰代码指南
发挥易读和易维护软件的好处 - 第一部分嗨,开发者们,清晰的代码是指编写易于阅读、理解和维护的软件代码。它是遵循一组原则和实践,优先考虑清晰性、简单性和一致性的代码。清...【详细内容】
2023-12-18  Search: Golang  点击:(125)  评论:(0)  加入收藏
如何优雅的组织Golang项目结构
一个Go项目的结构设计始终遵循Go语言的简洁高效理念。一个合理和良好的布局可以提高代码的可读性,简化依赖管理,并优化编译过程。像cmd、internal和docs这样的目录是标准Go项...【详细内容】
2023-12-06  Search: Golang  点击:(169)  评论:(0)  加入收藏
从 Discord 的做法中学习 — 使用 Golang 进行请求合并
正如你可能之前看到的,Discord去年发布了一篇有价值的文章,讨论了他们成功存储了数万亿条消息。虽然有很多关于这篇文章的YouTube视频和文章,但我认为这篇文章中一个名为“数据...【详细内容】
2023-11-24  Search: Golang  点击:(257)  评论:(0)  加入收藏
使用Golang进行自动化的20个库
Golang,也被称为Go,是一种静态类型的编译型编程语言,由Robert Griesemer,Rob Pike和Ken Thompson在Google设计。它在2009年推出,旨在解决其他编程语言在并发编程、垃圾回收和代码...【详细内容】
2023-11-23  Search: Golang  点击:(204)  评论:(0)  加入收藏
Golang 中的 Bytes 包详解之 Bytes.Buffer
上篇文章详细讲解了一次性密码 OTP 相关的知识,基于时间的一次性密码 TOTP 是 OTP 的一种实现方式。这种方法的优点是不依赖网络,因此即使在没有网络的情况下,用户也可以生成密...【详细内容】
2023-11-07  Search: Golang  点击:(258)  评论:(0)  加入收藏
聊聊Golang饱受争议的Error
一、error是什么?在C中,返回错误通过errno.h中的错误代码来表示,比如0代表No error,也就是没有错误;2代表No such file or directory,也就是找不到指定路径的文件或文件夹;5代表Inp...【详细内容】
2023-11-06  Search: Golang  点击:(254)  评论:(0)  加入收藏
使用示例和应用程序全面了解高效数据管理的Golang MySQL数据库
Golang,也被称为Go,已经成为构建强大高性能应用程序的首选语言。在处理MySQL数据库时,Golang提供了一系列强大的库,简化了数据库交互并提高了效率。在本文中,我们将深入探讨一些...【详细内容】
2023-10-29  Search: Golang  点击:(307)  评论:(0)  加入收藏
Golang中的强大Web框架
揭示Fiber在Go Web开发中的特点和优势在不断发展的Web开发领域中,选择正确的框架可以极大地影响项目的效率和成功。介绍一下Fiber,这是一款令人印象深刻的Golang(Go语言)Web框架...【详细内容】
2023-10-26  Search: Golang  点击:(258)  评论:(0)  加入收藏
▌简易百科推荐
宝藏级Go语言开源项目——教你自己动手开发互联网搜索引擎
DIYSearchEngine 是一个能够高速采集海量互联网数据的开源搜索引擎,采用 Go 语言开发。Github 地址:https://github.com/johnlui/DIYSearchEngine运行方法首先,给自己准备一杯...【详细内容】
2024-03-12  OSC开源社区    Tags:Go语言   点击:(18)  评论:(0)  加入收藏
Go Gin框架实现优雅地重启和停止
在Web应用程序中,有时候我们需要重启或停止服务器,无论是因为更新代码还是进行例行维护。在这种情景下,我们需要保证应用程序的可用性和数据的一致性。这就需要优雅地关闭和重...【详细内容】
2024-01-30  源自开发者  微信公众号  Tags:Go   点击:(67)  评论:(0)  加入收藏
如何让Go程序以后台进程或daemon方式运行
本文探讨了如何通过Go代码实现在后台运行的程序。最近我用Go语言开发了一个WebSocket服务,我希望它能在后台运行,并在异常退出时自动重新启动。我的整体思路是将程序转为后台...【详细内容】
2024-01-26  Go语言圈  微信公众号  Tags:Go程序   点击:(60)  评论:(0)  加入收藏
深入Go底层原理,重写Redis中间件实战
Go语言以其简洁、高效和并发性能而闻名,深入了解其底层原理可以帮助我们更好地利用其优势。在本文中,我们将探讨如何深入Go底层原理,以及如何利用这些知识重新实现一个简单的Re...【详细内容】
2024-01-25  547蓝色星球    Tags:Go   点击:(66)  评论:(0)  加入收藏
Go 内存优化与垃圾收集
Go提供了自动化的内存管理机制,但在某些情况下需要更精细的微调从而避免发生OOM错误。本文将讨论Go的垃圾收集器、应用程序内存优化以及如何防止OOM(Out-Of-Memory)错误。Go...【详细内容】
2024-01-15  DeepNoMind  微信公众号  Tags:Go   点击:(61)  评论:(0)  加入收藏
Go函数指针是如何让你的程序变慢的?
导读Go 语言的常规优化手段无需赘述,相信大家也能找到大量的经典教程。但基于 Go 的函数值问题,业界还没有太多深度讨论的内容分享。本文作者根据自己对 Go 代码的使用与调优...【详细内容】
2024-01-15  腾讯云开发者  微信公众号  Tags:Go函数   点击:(85)  评论:(0)  加入收藏
Go编程中调用外部命令的几种场景
在很多场合, 使用Go语言需要调用外部命令来完成一些特定的任务, 例如: 使用Go语言调用Linux命令来获取执行的结果,又或者调用第三方程序执行来完成额外的任务。在go的标准库...【详细内容】
2024-01-09  suntiger    Tags:Go编程   点击:(100)  评论:(0)  加入收藏
Go 语言不支持并发读写 Map,为什么?
Go语言的map类型不支持并发读写的主要原因是并发读写会导致数据竞态(data race),这意味着多个 goroutine 可能同时访问并修改同一个 map,从而引发不确定的结果。在Go语言的设计...【详细内容】
2024-01-05  Go语言圈  微信公众号  Tags:Go 语言   点击:(76)  评论:(0)  加入收藏
Go微服务入门到容器化实践
Go微服务入门到容器化实践Go 是一门高效、现代化、快速增长的编程语言,非常适合构建 Web 应用程序。而 Docker 是一种轻量级的容器化技术,能够使得您的应用程序在任何地方运行...【详细内容】
2024-01-01  大雷家吃饭    Tags:Go微服务   点击:(61)  评论:(0)  加入收藏
你是否想知道如何应对高并发?Go语言为你提供了答案!
并发编程是当前软件领域中不可忽视的一个关键概念。随着CPU等硬件的不断发展,我们都渴望让我们的程序运行速度更快、更快。而Go语言在语言层面天生支持并发,充分利用现代CPU的...【详细内容】
2023-12-29  灵墨AI探索室  微信公众号  Tags:Go语言   点击:(107)  评论:(0)  加入收藏
站内最新
站内热门
站内头条