Hyperledger Fabric(2) - 源码分析之Config 配置模块的设计

2024-05-07 13:32

本文主要是介绍Hyperledger Fabric(2) - 源码分析之Config 配置模块的设计,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 背景

一个优秀的区块链开源性项目,配置参数的读取是必不可少的。通常来讲,项目会基于现有配置读取框架上进行开发。

本文主要从源码角度探索Fabric 的配置读取原理。

2.查找相关代码

从 Hyperledger Fabric(1) - 整体架构和源码结构 文章中可知/fabric/core目录是项目的核心代码区,所以从此处入手查找配置参数的读取模块是否在其中。

果不其然/fabric/core/config目录即是配置的代码,查看 /fabric/core/config/config.go 代码内容可发现Fabric 的配置系统主要运用第三方包 viper。

viper可以对系统环境变量、 yaml/json 等格式的配置文件甚至是远程配置进行读取和设置 ,并可以在不重启服务的情况下动态设置新的配 项的值并使之实时生效。是一个专门处理配置的解决方案,拥有眼镜蛇的称号,和命令行的库 cobra 出自同一个开发者 spfl3 之手。

3. 源码分析

3.1 读取配置文件主模块

/fabric/core/config/config.go 中代码量不多,主要看InitViper函数,具体查看以下中文注释

func InitViper(v *viper.Viper, configName string) error {var altPath = os.Getenv("FABRIC_CFG_PATH")// 首先判断环境变量 FABRIC_CFG_PATH 是否有值, 如果有值说明是用户人为定义FABRIC 配置文件的路径if altPath != "" {// If the user has overridden the path with an envvar, its the only path// we will consider//判断路径是否存在if !dirExists(altPath) {return fmt.Errorf("FABRIC_CFG_PATH %s does not exist", altPath)}//如果路径存在,将FABRIC_CFG_PATH 内容添加作为配置文件的路径AddConfigPath(v, altPath)/*若没有定义该环境变量的值 ,则用代码添加两个路径作为搜索配置文件的路径1.首先将当前路径作为搜索配置文件的路径2.如果OfficialPath = "/etc/hyperledger/fabric"存在,则使用此路径作为搜索配置文件的路径*/} else {// If we get here, we should use the default paths in priority order://// *) CWD// *) /etc/hyperledger/fabric// CWDAddConfigPath(v, "./")// And finally, the official pathif dirExists(OfficialPath) {AddConfigPath(v, OfficialPath)}}/*调用 SetConfigName()设置配置文件名,所指的的配置文件名 configName 是由参数传递进来的*/// Now set the configuration file.if v != nil {v.SetConfigName(configName)} else {viper.SetConfigName(configName)}return nil
}

另外需要注意的是 InitViper 第一个参数 *viper.Viper 。在InitViper 函数中,无论是添加搜索路径(使用的是 AddConfigPath 函数),还是设置要搜索的配置文件名( SetConfigName函数),都分为全局的 viper 和特定的 viper(也就是参数最终由viper.AddConfigPathviper. SetConfigName 完成是全局的; 由 v.AddConfigPathv.SetConfigName 完成则是特定的。

这样可以很方便地初始化需要单独使用 viper的模块 。如下篇文章将要分析的 orderer 模块,即是采用单独的viper来进行配置。

3.2 peer命令搜索路径和配置文件

/fabric/cmd从命令方式看,大概率是此项目主题或者工具的命令启动代码,从此入口可以找到/fabric/cmd/peer/main.go文件,代码如下,具体查看以下中文注释

// The main command describes the service and
// defaults to printing the help message.
var mainCmd = &cobra.Command{Use: "peer"}func main() {// For environment variables.//通过viper设置前缀为 common.CmdRoot = "core"环境变量前缀名viper.SetEnvPrefix(common.CmdRoot)//通过viper获取所有的环境变量,如果设置过了前缀则会自动补全前缀名viper.AutomaticEnv()replacer := strings.NewReplacer(".", "_")viper.SetEnvKeyReplacer(replacer)// Define command-line flags that are valid for all peer commands and// subcommands.mainFlags := mainCmd.PersistentFlags()mainFlags.String("logging-level", "", "Legacy logging level flag")viper.BindPFlag("logging_level", mainFlags.Lookup("logging-level"))mainFlags.MarkHidden("logging-level")cryptoProvider := factory.GetDefault()mainCmd.AddCommand(version.Cmd())mainCmd.AddCommand(node.Cmd())mainCmd.AddCommand(chaincode.Cmd(nil, cryptoProvider))mainCmd.AddCommand(channel.Cmd(nil))mainCmd.AddCommand(lifecycle.Cmd(cryptoProvider))// On failure Cobra prints the usage message and error string, so we only// need to exit with a non-0 statusif mainCmd.Execute() != nil {os.Exit(1)}
}
3.3 orderer命令如何搜索、读取配置文件

同样的,如果看/fabric/cmd/orderer/main.go文件,具体查看以下中文注释

// Main is the entry point of orderer process
func Main() {fullCmd := kingpin.MustParse(app.Parse(os.Args[1:]))// "version" commandif fullCmd == version.FullCommand() {fmt.Println(metadata.GetVersionInfo())return}//配置文件加载入口conf, err := localconfig.Load()if err != nil {logger.Error("failed to parse config: ", err)os.Exit(1)}initializeLogging()prettyPrintStruct(conf)

/fabric/orderer/common/loadconfig/config.go,具体查看以下中文注释

// Load parses the orderer YAML file and environment, producing
// a struct suitable for config use, returning error on failure.
func Load() (*TopLevel, error) {//新建独立的viperconfig := viper.New()//调用 InitViper函数,configName 为配置文件名`orderer`coreconfig.InitViper(config, "orderer")  //`coreconfig`包就是`/fabric/core/config`包//通过viper设置前缀为 Prefix = "ORDERER"环境变量前缀名config.SetEnvPrefix(Prefix)//通过viper获取所有的环境变量,如果设置过了前缀则会自动补全前缀名config.AutomaticEnv()replacer := strings.NewReplacer(".", "_")config.SetEnvKeyReplacer(replacer)//读取配置文件if err := config.ReadInConfig(); err != nil {return nil, fmt.Errorf("Error reading configuration: %s", err)}var uconf TopLevelif err := viperutil.EnhancedExactUnmarshal(config, &uconf); err != nil {return nil, fmt.Errorf("Error unmarshaling config into struct: %s", err)}uconf.completeInitialization(filepath.Dir(config.ConfigFileUsed()))return &uconf, nil
}

4. 继续从源码分析其它模块

请见下文
Hyperledger Fabric(3) - 源码分析之Peer启动流程

这篇关于Hyperledger Fabric(2) - 源码分析之Config 配置模块的设计的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/967485

相关文章

SpringBoot中六种批量更新Mysql的方式效率对比分析

《SpringBoot中六种批量更新Mysql的方式效率对比分析》文章比较了MySQL大数据量批量更新的多种方法,指出REPLACEINTO和ONDUPLICATEKEY效率最高但存在数据风险,MyB... 目录效率比较测试结构数据库初始化测试数据批量修改方案第一种 for第二种 case when第三种

解决1093 - You can‘t specify target table报错问题及原因分析

《解决1093-Youcan‘tspecifytargettable报错问题及原因分析》MySQL1093错误因UPDATE/DELETE语句的FROM子句直接引用目标表或嵌套子查询导致,... 目录报js错原因分析具体原因解决办法方法一:使用临时表方法二:使用JOIN方法三:使用EXISTS示例总结报错原

nginx 负载均衡配置及如何解决重复登录问题

《nginx负载均衡配置及如何解决重复登录问题》文章详解Nginx源码安装与Docker部署,介绍四层/七层代理区别及负载均衡策略,通过ip_hash解决重复登录问题,对nginx负载均衡配置及如何... 目录一:源码安装:1.配置编译参数2.编译3.编译安装 二,四层代理和七层代理区别1.二者混合使用举例

Java JDK1.8 安装和环境配置教程详解

《JavaJDK1.8安装和环境配置教程详解》文章简要介绍了JDK1.8的安装流程,包括官网下载对应系统版本、安装时选择非系统盘路径、配置JAVA_HOME、CLASSPATH和Path环境变量,... 目录1.下载JDK2.安装JDK3.配置环境变量4.检验JDK官网下载地址:Java Downloads

Linux下进程的CPU配置与线程绑定过程

《Linux下进程的CPU配置与线程绑定过程》本文介绍Linux系统中基于进程和线程的CPU配置方法,通过taskset命令和pthread库调整亲和力,将进程/线程绑定到特定CPU核心以优化资源分配... 目录1 基于进程的CPU配置1.1 对CPU亲和力的配置1.2 绑定进程到指定CPU核上运行2 基于

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串

Spring Boot spring-boot-maven-plugin 参数配置详解(最新推荐)

《SpringBootspring-boot-maven-plugin参数配置详解(最新推荐)》文章介绍了SpringBootMaven插件的5个核心目标(repackage、run、start... 目录一 spring-boot-maven-plugin 插件的5个Goals二 应用场景1 重新打包应用

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数

Java中读取YAML文件配置信息常见问题及解决方法

《Java中读取YAML文件配置信息常见问题及解决方法》:本文主要介绍Java中读取YAML文件配置信息常见问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录1 使用Spring Boot的@ConfigurationProperties2. 使用@Valu

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期