多环境下的配置管理方案

前端之家收集整理的这篇文章主要介绍了多环境下的配置管理方案前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。

原文发布在博客

在开发中,我们需要面对各种各样的环境,开发环境、测试环境、生产环境……

并且,各个环境的参数和配置各不相同,比如数据库连接,服务器配置等。我们怎样在不同环境中调用正确的配置?

通过配置文件

这是一种常见的思路,通过创建多个配置文件,但根据命名区分,比如开发环境为develop-app.conf,测试环境为testing-app.conf,生产环境为production-app.conf

我们通过在系统中设置环境变量export ENV_MODE=develop等等。在读取配置文件时,根据环境变量读取响应的配置文件

这个方式易于使用,深得大家喜爱。但这个方案在集群扩大的一定程度时,会遇到一下几个主要问题:

  • 假如有30~40个微服务需要连接数据库运行,这个量级在中小型团队中很常见了,如果我们需要更改数据库密码,我们不得不将数十个project逐个进行更新,非常不灵活。

  • 代码与配置掺杂在一起,代码是许多开发人员都可以看到的,也很容易泄露,而生产环境的各种秘钥应该只有少数人有权限能看到。这对系统的安全有重大影响。

  • 对于大量相同的配置(比如数据库配置),逻辑上我们应该存放在同一个地方,保证只有唯一可靠的数据来源。

对于这些问题,我们认为配置应该集中化管理。

集中化管理带来以下好处:

  • 各个服务间相同的配置只需要维护一分数据,保证唯一性
  • 各个环境的配置环境实现权限隔离,少数人拥有查看生产环境配置的权限
  • 更改配置将变得简单,不影响服务本身

最简单的方案就是存储在redis中。KV的存储方式天然适合关联配置文件。但要完整的使用整个方案,需要做一些准备。

集中式配置管理

我们的基本思路是:将配置文件的值替换为占位符,在系统启动时,相应的工具将根据占位符到redis中查询到实际的值,替换回配置文件

最初的配置文件是这样:

{
  "database_host":"127.0.0.1","database_port":3306
}

现在我们的配置文件变成了这样:

{
  "database_host":"{{redis_hget "global.MysqL" "host"}}","database_port":{{redis_hget "global.MysqL" "port"}}
}

读取配置

在启动时,我们通过这个工具:https://github.com/gogap/env_json

这样读取配置文件

func main() {
    data,_ := IoUtil.ReadFile("./db.conf")
    dbConf := DBConfig{}
    if err := env_json.Unmarshal(data,&dbConf); err != nil {
        fmt.Print(err)
        return
    }
    fmt.Println(dbConf)
}

这个工具,默认从/etv/env_string.conf读取redis的配置信息,当然你可以更改,更多细节参看说明文档。

在这个过程中,env_json首先会从/etv/env_string.conf读取到redis的配置信息。

典型的/etv/env_string.conf内容如下

{
"storages": [{
        "engine": "redis","options": {
            "db": 0,"password": "","pool_size": 10,"address": "127.0.0.1:6379"
        }
    }]
}

连接上redis后,以上面的例子来说,将执行hget global.MysqL host以及hget global.MysqL port,将取到的值通过模板替换,更新到配置文件中,得到一个正常的json文本,剩下的就是通过json库把json内容解码到结构体中。

到目前为止,我们实现了从redis中读取并替换配置,那么我们写入配置的时候呢?

假如我们有数十个服务,我们难道需要逐个去redis中设置吗?我们怎样把这个流程自动化?

写入配置

我们需要另一个工具:env_sync

我们存储配置文件其实是一个具体的git工程,比如开发环境是develop_env,生产环境是production_env,开发人员都可以编辑develop_env这个工程,少数人可以编辑production_env。

工程里的内容什么呢?

我们约定了这样的目录结构

develop_env
	global.MysqL    //this is folder
         data       //this is file
    components.accounts
    	 data

在工程中,有一系列的文件夹,文件夹中有一个叫data的文件。这样的目录结构会被env_sync识别到,并转化成一系列的redis命令。

假如global.MysqL文件夹下的data文件内容

{
  "host":"127.0.0.1","port":3306
}

转化出来的命令是:

hset global.MysqL host 127.0.0.1
hset global.MysqL port 3306

此过程与读取过程正好相反,同样的,env_sync也是从/etc/env_strings.conf读取配置信息。与读取工具保持了统一。

总结

整体来看我们需要做几个工作

再结合自动化部署工具,每次配置文件有更新时,我们就在线上环境自动同步到redis。

更多

还有一种需求时,配置文件会动态变化,而我们不想重启服务就读取到配置文件,那你需要https://github.com/gogap/redconf

这个工具可以实现对redis中数据的检测,如果数据发生变化,会触发回调,应用可以得到变化前后的值。

原文链接:https://www.f2er.com/go/189703.html

猜你在找的Go相关文章