使用 Golang 构建可维护的 SQL 查询

2024-08-16 0 1,053

使用 Golang 构建可维护的 SQL 查询

任何使用 sql 查询的应用程序都可以受益于使用查询生成器来提高代码的可读性、可维护性和安全性。事实上,golang 中有许多不同的库可以做到这一点。在 vaunt,我们尝试了许多不同的选择,最后决定自己创建一个。最终,我们想要一些安全的东西,并提供变量替换来防止 sql 注入,同时仍然可读并且能够有条件语句。因此,我们创建了一个名为 tqla 的新库,并于去年年底发布并宣布。您可以在这篇文章中阅读更多相关信息。

在构建 tqla 之前,我们主要使用 squirrel 来构建 sql 查询逻辑——我们强烈推荐它。我们仍然在某些领域使用 squirrel,但已逐渐开始用 tqla 替换和实现新的查询构建逻辑。我们发现在许多实例中,tqla 提高了我们维护代码和修复使用其他语句生成器时遇到的问题的能力。

现实世界用例

在 vaunt,我们最近进行了从 cockroachdb 到 tidb 的数据库迁移。虽然 cockroachdb 高性能且可靠,但我们最终决定添加到我们的技术堆栈中以支持 olap 数据库。这样做的需要是支持我们对开源社区洞察产品的分析工作量。为了保持较小的技术足迹,我们决定继续使用 tidb 并利用该数据库的 htap 架构。

cockroachdb 与 postgresql 很大程度上兼容,我们的许多 sql 查询都使用 postgresql 语法。要切换到 tidb,我们必须更改一些表并更新查询以使用 Mysql 语法。在迁移过程中的一些位置,我们发现我们不正确地使用条件查询构建语句,并且缺乏适当的测试来发现语句生成不正确。

立即学习“go语言免费学习笔记(深入)”;

示范

在 squirrel 的自述文件中,有一个示例说明如何使用条件查询构建来更新带有可选过滤器的语句:

1

2

3

if len(q) > 0 {

    users = users.Where("name like ?", fmt.sprint("%", q, "%"))

}

这是一个真实但简化的示例,说明我们如何更新其中一个查询以有条件连接表并添加可选过滤器:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

PSql := squirrel.statementbuilder.placeholderfORMat(squirrel.question)

statementbuilder := psql.select(`i.id`).

from("vaunt.installations i").

where(`entity_name = ?`, name)

if len(provider) > 0 {

    statementbuilder.where(`provider = ?`, provider)

}

if len(repo) > 0 {

    statementbuilder.join(`repositories as r on JSon_contains(i.repositories, concat('["', r.id, '"]'))`)

    statementbuilder.where(`r.name = ?`, repo)

}

你能发现代码的问题吗?如果没有,请不要担心——在我们运行测试之前,我们自己的代码审查也会忽略这一点。

这里的问题是我们忘记使用构建器函数的结果更新语句构建器。例如,提供者条件过滤器应改为:

1

2

3

if len(provider) > 0 {

    statementbuilder = statementbuilder.where(`provider = ?`, provider)

}

这是一个相对简单的错误,通过足够的测试用例很容易发现,但由于它不是技术上无效的代码,因此可能需要一些时间才能立即意识到发生了什么。

此设置的另一个可读性问题是条件连接与初始 select 语句是分开的。我们可以重新组织构建器,将每个部分放在它应该去的地方,但这需要多次重复的条件语句检查,并且仍然会遇到一些可读性问题。

使用tqla

上面使用 squirrel 的演示已被重写,tqla 中的等效项如下所示:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

t, err := tqla.new(tqla.withplaceholder(tqla.question))

if err != nil {

    return nil, err

}

query, args, err := t.compile(`

    select i.id

    from vaunt.installations as i

    {{ if .repo }}

    join vaunt.repositories as r on JSON_contains(i.repositories, concat('["', r.id, '"]'), '$')

    {{ end }}

    where entity_name = {{ .name}}

    {{ if .provider }}

    and i.provider = {{ .provider }}

    {{ end }}

    {{ if .repo }}

    and r.name = {{ .repo }}

    {{ end }}

    `, data)

if err != nil {

    return nil, err

}

如您所见,tqla 的模板语法使得合并条件子句变得非常简单。 tqla 自动用指定的占位符替换我们设置的变量,并提供我们可以与 sql 驱动程序一起使用来执行语句的参数。

与 squirrel 类似,这种语句构建方法很容易测试,因为我们可以创建不同的数据对象集来传递给模板构建器并验证输出。

您可以看到,我们可以轻松地在最适合的位置添加查询的条件部分。例如,这里我们在 from 语句之后直接有一个条件 join,尽管我们仍然有多个条件检查,但它并没有使模板过于复杂。

自定义功能

另一个很好的 tqla 功能有助于提高 sql 构建器的可维护性,是能够定义我们可以在模板中使用的自定义函数来抽象一些转换逻辑。

这是我们如何使用函数将 golang 的 time.time 值转换为 sql.nulltime 的示例,以便我们可以对数据对象进行插入,而无需事先转换:

1

2

3

4

5

6

7

8

9

10

11

12

13

funcs := template.funcmap{

    "time": func(t time.time) sql.nulltime {

        if t.iszero() {

            return sql.nulltime{valid: false}

        }

        return sql.nulltime{time: t, valid: true}

    },

}

t, err := tqla.new(tqla.withplaceholder(tqla.question), tqla.withfuncmap(funcs))

if err != nil {

    return err

}

通过在 tqla funcs 映射中定义此函数,我们现在可以通过向其提供来自数据对象(即 time.time 字段)的参数来在查询模板中自由使用它。我们甚至可以在同一个模板中使用不同的字段多次调用这个函数。

这是一个简化的示例:

1

2

3

4

5

6

7

8

9

statement, args, err := t.Compile(`

    INSERT INTO events

        (name, created_at, merged_at, closed_at)

    VALUES (

        {{ .Name }},

        {{ time .CreatEDAt }},

        {{ time .MergedAt }},

        {{ time .ClosedAt }}

    )`, eventdata)

结论

总之,我们相信使用 tqla 可以帮助提高查询构建逻辑的可维护性,同时为创建动态查询提供一些强大的实用程序。模板结构的简单性允许清晰的代码可读性,并且可以更快地调试任何潜在的错误。

我们将 tqla 开源来共享这个库,希望它为其他希望以简单、可维护且安全的方式在许多不同类型的应用程序中构建 sql 查询的用户提供一个很好的选择。

如果您有兴趣,请查看存储库,如果它对您有任何帮助,请给它一个星星。请随时提出任何功能请求或错误报告!

我们始终乐于接受反馈和贡献。

为了了解未来的发展,请在 x 上关注我们或加入我们的 discord!

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

免责声明
1. 本站所有资源来源于用户上传和网络等,如有侵权请邮件联系本站整改team@lcwl.fun!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系本站工作人员处理!
6. 本站资源售价或VIP只是赞助,收取费用仅维持本站的日常运营所需!
7. 如遇到加密压缩包,请使用WINRAR解压,如遇到无法解压的请联系管理员!
8. 因人力时间成本问题,部分源码未能详细测试(解密),不能分辨部分源码是病毒还是误报,所以没有进行任何修改,大家使用前请进行甄别!
9.本站所有源码资源都是经过本站工作人员人工亲测可搭建的,保证每个源码都可以正常搭建,但不保证源码内功能都完全可用,源码属于可复制的产品,无任何理由退款!

网站搭建学习网 Go 使用 Golang 构建可维护的 SQL 查询 https://www.xuezuoweb.com/13687.html

常见问题
  • 本站所有的源码都是经过平台人工部署搭建测试过可用的
查看详情
  • 购买源码资源时购买了带主机的套餐是指可以享受源码和所选套餐型号的主机两个产品,在本站套餐里开通主机可享优惠,最高免费使用主机
查看详情

相关文章

发表评论
暂无评论
官方客服团队

为您解决烦忧 - 24小时在线 专业服务

Fa快捷助手
手机编程软件开发

在手机上用手点一点就能轻松做软件

去做软件
链未云主机
免备案香港云主机

开通主机就送域名的免备案香港云主机

去使用
链未云服务器
免备案香港云服务器

支持售后、超低价、稳定的免备案香港云服务器

去使用