# 为什么封装?
在开源盛行的今天,我们没有必要也“几乎”不可能造所有的轮子。 开源界有很多优秀的项目可供挑选,使用开源项目可以极大地降低开发成本、快速地提升开发效率,但天下没有免费的午餐,使用开源项目的同时也带来很多问题:
- 用法差异大、用法复杂,无法快速上手
- 错误处理、日志输出、指标上报、配置加载支持程度不同
- 功能集合过大,如不加约束的使用,无法统一用法
- 质量参差不齐,如不加约束的引用,类库的缺陷就会不受限的在组织内传导,修复的成本非常高
- 未遵守版本规范,到底用哪个版本?
- 相同功能多个项目,到底选哪个?
- 直接使用,替换成本高
# 怎么封装?
因此对于引入的常用的、重要的开源类库,我们需进行统一封装,尽量减少直接引用开源软件。封装规范如下:
- 实现配置注册、动静态加载
- 实现工厂方法,通过
Config
构建实例,而不是直接New
创建实例 - 实现配置动静态加载、指标上报、链路跟踪、统一日志,统一错误等功能
- 启停功能通过
Serve
,Shutdown
实现,统一由 box application 管理 - 重要的 type 起别名,避免使用时再 import 开源项目
type alias: type T = package_xx.T1
- 重要的 const、var,func 做绑定,避免使用时再 import 开源项目
var ErrClosed = package_xx.ErrClosed
const Nil = redis.Nil
var Func = package_xx.Func
- 重要的、性能敏感的模块需编写 Benchmark,了解性能指标
- 至少提供一个 Test 和 Example
- 避免使用
type embedding
,虽然方便,但将暴露过多细节 - 模块间依赖使用配置重定向,比如:模块A依赖模块B,在模块A的Config中增加字段指向模块B的配置路径,详情见RedisCache案例 (opens new window)
# 模块Layout
我们以 pkg/client/redis (opens new window) 为例,介绍一下推荐的模块文件布局方案。
> tree pkg/client/redis
> pkg/client/redis
> ├── alias.go // 别名定义
> ├── config.go // 配置声明、加载
> ├── default.go // 默认实例及其导出方法
> ├── metric.go // 指标上报
> ├── redis.go // 类库封装
> ├── redis_test.go // 单元测试
> ├── script.go // 其他文件
> └── testdata // 测试依赖数据
> ├── ci.yaml // GitHub Action依赖配置
> └── local.yaml // 开发环境 test 依赖配置
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
# GoLand Live Template
使用GoLand的童鞋可以创建Live Template,通过模版快速生成代码,详情见Live templates (opens new window)。
import (
"github.com/boxgo/box/pkg/config"
"github.com/boxgo/box/pkg/logger"
)
type (
$Type$ struct {
cfg *Config
}
)
func new$Type$(c *Config) *$Type$ {
return &$Type${
cfg: c,
}
}
type (
// Config 配置
Config struct {
path string
Key string
}
// OptionFunc 选项信息
OptionFunc func(*Config)
)
// WithKey 设置选项
func WithKey(x string) OptionFunc {
return func(options *Config) {
options.Key = x
}
}
// StdConfig 标准配置
func StdConfig(key string, optionFunc ...OptionFunc) *Config {
cfg := DefaultConfig(key)
for _, fn := range optionFunc {
fn(cfg)
}
if err := config.Scan(cfg); err != nil {
logger.Panicf("$Type$ load config error: %s", err)
}
return cfg
}
// DefaultConfig 默认配置
func DefaultConfig(key string) *Config {
return &Config{
path: "$Type$." + key,
}
}
// Build 构建实例
func (c *Config) Build() *$Type$ {
return new$Type$(c)
}
// Path 实例配置目录
func (c *Config) Path() string {
return c.path
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
# FAQ
# 加载没有报错,但是配置未加载成功。
Config
的Path()
是否正确Cofnig
反序列化的tag
是config
- 字段必须是导出字段(大写开头),如下:
type (
Config struct {
field0 string // 不可导出字段,小写开头
Field1 string `config:"field1" desc:"field1 description"` // 导出字段,大写开头
Field3 int `config:"minIdleConnCnt" desc:"field3 description"` // 导出字段,大写开头
}
)
// Path 实例配置目录
func (c *Config) Path() string {
return c.path
}
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
← 配置