当前位置:首页 > 文章列表 > Golang > Go问答 > 在连线依赖注入中创建每个提供者的记录器

在连线依赖注入中创建每个提供者的记录器

来源:stackoverflow 2024-04-18 12:33:34 0浏览 收藏

本篇文章主要是结合我之前面试的各种经历和实战开发中遇到的问题解决经验整理的,希望这篇《在连线依赖注入中创建每个提供者的记录器》对你有很大帮助!欢迎收藏,分享给更多的需要的朋友学习~

问题内容

我正在使用 github.com/google/wire 在我正在开发的开源示例项目中进行依赖项注入。

我在名为 interfaces 的包中有以下接口:

type loginservice interface {
    login(email, password) (*loginresult, error)
}

type jwtservice interface {
    generate(user *models.user) (*jwtgenerateresult, error)
    validate(tokenstring string) (*jwtvalidateresult, error)
}

type userdao interface {
    byemail(email string) (*models.user, error)
}

我的实现如下所示:

857188​​294282

我将在这里排除其他工厂函数和实现,因为它们看起来都非常相似。他们可能会返回错误或绝对正确。

我还有另一个有趣的工厂用于创建数据库连接,我将仅显示其接口而不是实现:

func connect(config interfaces.mysqlconfig) (*gorm.db, error) { /* ... */ }

现在,解决问题。在我的命令行入口点中,我正在创建一个记录器:

logger, err := zap.newdevelopment()

对于上面的每个工厂方法,我需要提供一个记录器,而不是相同的记录器实例,就像这些方法是按如下方式调用的:

logger, err := zap.newdevelopment()

// check err

db, err := database.connect(config)

// check err

userdao := dao.newuserdao(db, logger.named("dao.user"))
jwtservice, err := service.newjwtservice(jwtkey)

// check err

loginservice := service.newloginservice(jwtservice, userdao, logger.named("service.login"))

我的 wire.providerset 构造如下所示:

wire.NewSet(
    wire.Bind(new(interfaces.LoginService), new(*service.LoginServiceImpl)),
    wire.Bind(new(interfaces.JWTService), new(*service.JWTServiceImpl)),
    wire.Bind(new(interfaces.UserDao), new(*dao.UserDaoImpl)),
    service.NewLoginService,
    service.NewJWTService,
    dao.NewUserDao,
    database.Connect,
)

我已经阅读了用户指南、教程和最佳实践,但我似乎找不到一种方法将唯一的 zap.logger 路由到每个工厂方法,并路由随机 [32 ]byte 用于 jwt 服务。

由于我的根记录器不是在编译时创建的,并且每个工厂方法都需要自己独特的记录器,因此我如何告诉 wire 将这些实例绑定到相应的工厂方法?我很难了解如何将相同类型的自定义实例路由到不同的工厂方法。

总结:

wire 似乎倾向于在编译时执行所有操作,将依赖项注入配置存储在静态包级变量中。对于我的大多数用例来说,这是可以的。 对于我的其余用例,我需要在运行依赖项注入之前手动创建一些实例,并能够将各种 *zap.logger 实例路由到每个需要它的服务。 本质上,我需要有wire do services.newuserdao(connect(mysqlconfig), logger.named("dao.user"),但我不知道如何在wire中表达这一点,并在运行时与wire的编译时合并变量方法。

如何在 wire 中执行此操作?


正确答案


我必须按照 documentation 中的建议稍微改变一下我正在做的事情:

添加自定义类型

文档确实非常简洁,但我最终所做的是创建一堆类型:

type jwtkey [32]byte
type jwtservicelogger *zap.logger
type loginservicelogger *zap.logger
type userdaologger *zap.logger

更新生产者函数

我更新了我的生产者方法以接受这些类型,但不必更新我的结构:

// loginserviceimpl implements interfaces.loginservice
var _ interfaces.loginservice = (*loginserviceimpl)(nil)

type loginserviceimpl struct {
    dao interfaces.userdao
    jwt interfaces.jwtservice
    logger *zap.logger
}

func newloginservice(dao interfaces.userdao, jwt interfaces.jwtservice, 
        logger loginservicelogger) *loginserviceimpl {
    return &loginserviceimpl {
        dao: dao,
        jwt: jwt,
        logger: logger,
    }
}

以上部分是有道理的;给出不同的类型意味着 wire 需要弄清楚的事情更少。

创建注入器

接下来,我必须创建虚拟注入器,然后使用 wire 生成相应的 wire_gen.go。这并不容易,而且非常不直观。当遵循文档时,事情不断发生故障并给我带来非常无用的错误消息。

我有一个 cmd/ 软件包,我的 cli 入口点位于 cmd/serve/root.go 中,它从命令行作为 ./apiserve 运行。我在 cmd/serve/injectors.go 中创建了注入器函数,请注意 // +buildwireinject 和以下换行符需要通知 go 该文件用于代码生成而不是代码本身。

经过多次尝试和错误,我最终得到了以下代码:

// +build wireinject

package serve

import /*...*/

func initializeloginservice(
        config interfaces.mysqlconfig,
        jwtkey service.jwtkey,
        loginservicelogger service.loginservicelogger,
        jwtservicelogger service.jwtservicelogger,
        userdaologger service.userdaologger,
        databaselogger database.databaselogger,
    ) (interfaces.loginservice, error) {
    
    wire.build(
        // bind interfaces to implementations
        wire.bind(new(interfaces.loginservice), new(*service.loginserviceimpl)),
        wire.bind(new(interfaces.jwtservice), new(*service.jwtserviceimpl)),
        wire.bind(new(interfaces.userdao), new(*dao.userdao)),
        // services
        service.newloginservice,
        service.newjwtservice,
        // daos
        dao.newuserdao,
        // database
        database.connect,
    )

    return nil, nil
}

wire.bind 调用通知 wire 对于给定接口使用哪个实现,以便它知道返回 *loginserviceimplservice.newloginservice 应该用作 interfaces.loginservice

调用 wire.build 中的其余实体只是工厂函数。

将值传递给注入器

我遇到的问题之一是我试图将值传递到 wire.build like the documentation describes

这就是让我困惑的地方;听起来你在尝试运行注入器时只能真正使用常量值,但是 there are two lines in the docs in the "injectors" section

这些行附有以下代码:

func initializebaz(ctx context.context) (foobarbaz.baz, error) {
    wire.build(foobarbaz.megaset)
    return foobarbaz.baz{}, nil
}

这就是我错过的,也是导致我在这方面浪费大量时间的原因。 context.context 似乎没有在这段代码中传递到任何地方,而且它是一种常见类型,所以我只是耸耸肩,没有从中学习。

我定义了注入器函数来获取 jwt 键、mysql 配置和记录器类型的参数:

func initializeloginservice(
        config interfaces.mysqlconfig,
        jwtkey service.jwtkey,
        loginservicelogger service.loginservicelogger,
        jwtservicelogger service.jwtservicelogger,
        userdaologger service.userdaologger,
        databaselogger database.databaselogger,
    ) (interfaces.loginservice, error) {
    // ...
    return nil, nil
}

然后,我尝试将它们注入到 wire.build 中:

wire.Build(
    // ...
    wire.Value(config),
    wire.Value(jwtKey),
    wire.Value(loginServiceLogger),
    // ...
)

当我尝试运行 wire 时,它抱怨这些类型被定义了两次。我对这种行为感到非常困惑,但最终了解到 wire 自动将所有函数参数发送到 wire.build

再一次:wire 自动将所有注入器功能参数发送到 wire.build

这对我来说并不直观,但我经历了惨痛的教训才知道这就是 wire 的工作方式。

摘要

wire 没有提供一种方法来区分其依赖注入系统中相同类型的值。因此,您需要用类型定义包装这些简单类型,让 wire 知道如何路由它们,因此不要使用 [32]byte,而是 type jwtkey [32]byte

要将实时值注入到 wire.build 调用中,只需更改注入器函数签名以将这些值包含在函数参数中,wire 就会自动将它们注入到 wire.build 中。

运行 cd pkg/my/package &&wire 在该目录中为您定义的注入器创建 wire_gen.go。完成此操作后,以后对 gogenerate 的调用将在发生更改时自动更新 wire_gen.go

我已将 wire_gen.go 文件签入到我的版本控制系统 (vcs)(即 git)中,由于这些生成的构建工件,这感觉很奇怪,但这似乎是通常完成此操作的方式。排除 wire_gen.go 可能会更有利,但如果这样做,您需要找到包含带有 // +buildwireinject 标头的文件的每个包,在该目录中运行 wire,然后 go 生成 可以肯定的是。

希望这能够澄清 wire 处理实际值的方式:使用类型包装器使它们类型安全,然后只需将它们传递给您的注入器函数,wire 就会完成剩下的工作。

本篇关于《在连线依赖注入中创建每个提供者的记录器》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
Java并行编程中线程优先级的设置与管理Java并行编程中线程优先级的设置与管理
上一篇
Java并行编程中线程优先级的设置与管理
国内应用厂商加速鸿蒙原生应用开发,<植物大战僵尸2>鸿蒙版即将上线
下一篇
国内应用厂商加速鸿蒙原生应用开发,<植物大战僵尸2>鸿蒙版即将上线
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 笔灵AI生成答辩PPT:高效制作学术与职场PPT的利器
    笔灵AI生成答辩PPT
    探索笔灵AI生成答辩PPT的强大功能,快速制作高质量答辩PPT。精准内容提取、多样模板匹配、数据可视化、配套自述稿生成,让您的学术和职场展示更加专业与高效。
    24次使用
  • 知网AIGC检测服务系统:精准识别学术文本中的AI生成内容
    知网AIGC检测服务系统
    知网AIGC检测服务系统,专注于检测学术文本中的疑似AI生成内容。依托知网海量高质量文献资源,结合先进的“知识增强AIGC检测技术”,系统能够从语言模式和语义逻辑两方面精准识别AI生成内容,适用于学术研究、教育和企业领域,确保文本的真实性和原创性。
    41次使用
  • AIGC检测服务:AIbiye助力确保论文原创性
    AIGC检测-Aibiye
    AIbiye官网推出的AIGC检测服务,专注于检测ChatGPT、Gemini、Claude等AIGC工具生成的文本,帮助用户确保论文的原创性和学术规范。支持txt和doc(x)格式,检测范围为论文正文,提供高准确性和便捷的用户体验。
    38次使用
  • 易笔AI论文平台:快速生成高质量学术论文的利器
    易笔AI论文
    易笔AI论文平台提供自动写作、格式校对、查重检测等功能,支持多种学术领域的论文生成。价格优惠,界面友好,操作简便,适用于学术研究者、学生及论文辅导机构。
    50次使用
  • 笔启AI论文写作平台:多类型论文生成与多语言支持
    笔启AI论文写作平台
    笔启AI论文写作平台提供多类型论文生成服务,支持多语言写作,满足学术研究者、学生和职场人士的需求。平台采用AI 4.0版本,确保论文质量和原创性,并提供查重保障和隐私保护。
    41次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码