五步助你成为优秀的Docker代码贡献者

云计算
开源渐成主流,越来越多的开发者想参与开源社区。而时下最火热的Docker也许就是开发者入手开源项目的最好选择,它不仅是目前最流行的开源项目之一,而且在提交Issue方面的文档和流程都是目前我见过的开源项目里最好的。本文主要介绍了如何入手开源项目,一些小经验和小工具,一起来学习。

开源渐成主流,越来越多的开发者想参与开源社区。而时下最火热的Docker也许就是开发者入手开源项目的***选择,它不仅是目前***的开源项目之一,而且在提交Issue方面的文档和流程都是目前我见过的开源项目里***的。本文主要介绍了如何入手开源项目,一些小经验和小工具,一起来学习。

成为一个流行开源项目(如Docker)的贡献者有如下好处:

你可以参与改进很多人都在使用的项目,以此来获得认同感;

你可以与开源社区中的那些聪明绝顶的人通力合作;

你可以通过参与理解和改进这个项目来使自己成为一名更加出色的程序员。

但是,从一个新的基准代码(codebase)入手绝对是一件恐怖的事情。目前,Docker已经有相当多的代码了,哪怕是修复一个小问题,都需要阅读大量的代码,并理解这些部分是如何组合在一起的。

不过,它们也并不如你想象的那么困难。你可以根据Docker的贡献者指南来完成环境的配置。然后按照如下5个简单的步骤,配合相关的代码片段来深入代码基。你所历练的这些技能,都将会在你的编程生涯的每个新项目中派上用场。那么还等什么,我们这就开始。

步骤1:从'func main()'开始

正如一句古话所述,从你知道的开始。如果你和大部分Docker用户一样,你可能主要使用Docker CLI。因此,让我们从程序的入口开始:‘main’函数。

此处为本文的提示,我们将会使用一个名为Sourcegraph的站点,Docker团队就使用它完成在线检索和代码浏览,和你使用智能IDE所做的差不多。建议在阅读本文时,打开Sourcegraph放在一边,以更好地跟上文章的进度。

在Sourcegraph站点,让我们搜索Docker仓库中的‘func main()’。

 

[[137576]]

我们正在寻找对应‘docker’命令的‘main’函数,它是‘docker/docker/docker.go’中的一个文件。点击搜索结果,我们会跳到其定义(如下所示)。花一点时间浏览一下这个函数:

 

  1. func main() { 
  2. if reexec.Init() { 
  3.     return 
  4.  
  5. // Set terminal emulation based on platform as required. 
  6. stdin, stdout, stderr := term.StdStreams() 
  7.  
  8. initLogging(stderr) 
  9.  
  10. flag.Parse() 
  11. // FIXME: validate daemon flags here 
  12.  
  13. if *flVersion { 
  14.     showVersion() 
  15.     return 
  16.  
  17. if *flLogLevel != "" { 
  18.     lvl, err := logrus.ParseLevel(*flLogLevel) 
  19.     if err != nil { 
  20.         logrus.Fatalf("Unable to parse logging level: %s", *flLogLevel) 
  21.     } 
  22.     setLogLevel(lvl) 
  23. else { 
  24.     setLogLevel(logrus.InfoLevel) 
  25.  
  26. // -D, --debug, -l/--log-level=debug processing 
  27. // When/if -D is removed this block can be deleted 
  28. if *flDebug { 
  29.     os.Setenv("DEBUG""1"
  30.     setLogLevel(logrus.DebugLevel) 
  31.  
  32. if len(flHosts) == 0 { 
  33.     defaultHost := os.Getenv("DOCKER_HOST"
  34.     if defaultHost == "" || *flDaemon { 
  35.         // If we do not have a host, default to unix socket 
  36.         defaultHost = fmt.Sprintf("unix://%s", api.DEFAULTUNIXSOCKET) 
  37.     } 
  38.     defaultHost, err := api.ValidateHost(defaultHost) 
  39.     if err != nil { 
  40.         logrus.Fatal(err) 
  41.     } 
  42.     flHosts = append(flHosts, defaultHost) 
  43.  
  44. setDefaultConfFlag(flTrustKey, defaultTrustKeyFile) 
  45.  
  46. if *flDaemon { 
  47.     if *flHelp { 
  48.         flag.Usage() 
  49.         return 
  50.     } 
  51.     mainDaemon() 
  52.     return 
  53.  
  54. if len(flHosts) > 1 { 
  55.     logrus.Fatal("Please specify only one -H"
  56. protoAddrParts := strings.SplitN(flHosts[0], "://"2
  57.  
  58. var ( 
  59.     cli       *client.DockerCli 
  60.     tlsConfig tls.Config 
  61. tlsConfig.InsecureSkipVerify = true 
  62.  
  63. // Regardless of whether the user sets it to true or false, if they 
  64. // specify --tlsverify at all then we need to turn on tls 
  65. if flag.IsSet("-tlsverify") { 
  66.     *flTls = true 
  67.  
  68. // If we should verify the server, we need to load a trusted ca 
  69. if *flTlsVerify { 
  70.     certPool := x509.NewCertPool() 
  71.     file, err := ioutil.ReadFile(*flCa) 
  72.     if err != nil { 
  73.         logrus.Fatalf("Couldn't read ca cert %s: %s", *flCa, err) 
  74.     } 
  75.     certPool.AppendCertsFromPEM(file) 
  76.     tlsConfig.RootCAs = certPool 
  77.     tlsConfig.InsecureSkipVerify = false 
  78.  
  79. // If tls is enabled, try to load and send client certificates 
  80. if *flTls || *flTlsVerify { 
  81.     _, errCert := os.Stat(*flCert) 
  82.     _, errKey := os.Stat(*flKey) 
  83.     if errCert == nil && errKey == nil { 
  84.         *flTls = true 
  85.         cert, err := tls.LoadX509KeyPair(*flCert, *flKey) 
  86.         if err != nil { 
  87.             logrus.Fatalf("Couldn't load X509 key pair: %q. Make sure the key is encrypted", err) 
  88.         } 
  89.         tlsConfig.Certificates = []tls.Certificate{cert} 
  90.     } 
  91.     // Avoid fallback to SSL protocols < TLS1.0 
  92.     tlsConfig.MinVersion = tls.VersionTLS10 
  93.  
  94. if *flTls || *flTlsVerify { 
  95.     cli = client.NewDockerCli(stdin, stdout, stderr, *flTrustKey, protoAddrParts[0], protoAddrParts[1], &tlsConfig) 
  96. else { 
  97.     cli = client.NewDockerCli(stdin, stdout, stderr, *flTrustKey, protoAddrParts[0], protoAddrParts[1], nil) 
  98.  
  99. if err := cli.Cmd(flag.Args()...); err != nil { 
  100.     if sterr, ok := err.(*utils.StatusError); ok { 
  101.         if sterr.Status != "" { 
  102.             logrus.Println(sterr.Status) 
  103.         } 
  104.         os.Exit(sterr.StatusCode) 
  105.     } 
  106.     logrus.Fatal(err) 
  107. }  

 

在‘main’函数的顶部,我们看了许多与日志配置,命令标志读取以及默认初始化相关的代码。在底部,我们发现了对『client.NewDockerCli』的调用,它似乎是用来负责创建结构体的,而这个结构体的函数则会完成所有的实际工作。让我们来搜索『NewDockerCli』。

步骤2:找到核心部分

在很多的应用和程序库中,都有1到2个关键接口,它表述了核心功能或者本质。让我们尝试到达这个关键部分。

点击‘NewDockerCli’的搜索结果,我们会到达函数的定义。由于我们感兴趣的只是这个函数所返回的结构体——「DockerCli」,因此让我们点击返回类型来跳转到其定义。

 

  1. func NewDockerCli(in io.ReadCloser, out, err io.Writer, keyFile string, proto, addr string, tlsConfig *tls.Config) *DockerCli { 
  2. var ( 
  3.     inFd          uintptr 
  4.     outFd         uintptr 
  5.     isTerminalIn  = false 
  6.     isTerminalOut = false 
  7.     scheme        = "http" 
  8.  
  9. if tlsConfig != nil { 
  10.     scheme = "https" 
  11. if in != nil { 
  12.     inFd, isTerminalIn = term.GetFdInfo(in
  13.  
  14. if out != nil { 
  15.     outFd, isTerminalOut = term.GetFdInfo(out
  16.  
  17. if err == nil { 
  18.     err = out 
  19.  
  20. // The transport is created here for reuse during the client session 
  21. tr := &http.Transport{ 
  22.     TLSClientConfig: tlsConfig, 
  23.  
  24. // Why 32? See issue 8035 
  25. timeout := 32 * time.Second 
  26. if proto == "unix" { 
  27.     // no need in compressing for local communications 
  28.     tr.DisableCompression = true 
  29.     tr.Dial = func(_, _ string) (net.Conn, error) { 
  30.         return net.DialTimeout(proto, addr, timeout) 
  31.     } 
  32. else { 
  33.     tr.Proxy = http.ProxyFromEnvironment 
  34.     tr.Dial = (&net.Dialer{Timeout: timeout}).Dial 
  35.  
  36. return &DockerCli{ 
  37.     proto:         proto, 
  38.     addr:          addr, 
  39.     in:            in
  40.     out:           out
  41.     err:           err, 
  42.     keyFile:       keyFile, 
  43.     inFd:          inFd, 
  44.     outFd:         outFd, 
  45.     isTerminalIn:  isTerminalIn, 
  46.     isTerminalOut: isTerminalOut, 
  47.     tlsConfig:     tlsConfig, 
  48.     scheme:        scheme, 
  49.     transport:     tr, 
  50. }  

 

点击『DockerCli』将我们带到了它的定义。向下滚动这个文件,我们可以看到它的方法,‘getMethod’,‘Cmd’,‘Subcmd’和‘LoadConfigFile’。其中,‘Cmd’ 值得留意。它是唯一一个包含docstring的方法,而docstring则表明它是执行每条Docker命令的核心方法。

步骤3:更进一步

既然我们已经找到了‘DockerCli’,这个Docker客户端的核心‘控制器’,接下来让我们继续深入,了解一条具体的Docker命令是如何工作的。让我们放大‘docker build’部分的代码。

 

  1. type DockerCli struct { 
  2. proto      string 
  3. addr       string 
  4. configFile *registry.ConfigFile 
  5. in         io.ReadCloser 
  6. out        io.Writer 
  7. err        io.Writer 
  8. keyFile    string 
  9. tlsConfig  *tls.Config 
  10. scheme     string 
  11. // inFd holds file descriptor of the client's STDIN, if it's a valid file 
  12. inFd uintptr 
  13. // outFd holds file descriptor of the client's STDOUT, if it's a valid file 
  14. outFd uintptr 
  15. // isTerminalIn describes if client's STDIN is a TTY 
  16. isTerminalIn bool 
  17. // isTerminalOut describes if client's STDOUT is a TTY 
  18. isTerminalOut bool 
  19. transport     *http.Transport 
  20. }  

 

阅读‘DockerCli.Cmd’的实现可以发现,它调用了‘DockerCli.getMethod’方法来执行每条Docker命令所对应的函数。

 

  1. func (cli *DockerCli) Cmd(args ...string) error { 
  2. if len(args) > 1 { 
  3.     method, exists := cli.getMethod(args[:2]...) 
  4.     if exists { 
  5.         return method(args[2:]...) 
  6.     } 
  7. if len(args) > 0 { 
  8.     method, exists := cli.getMethod(args[0]) 
  9.     if !exists { 
  10.         fmt.Fprintf(cli.err, "docker: '%s' is not a docker command. See 'docker --help'.\n", args[0]) 
  11.         os.Exit(1
  12.     } 
  13.     return method(args[1:]...) 
  14. return cli.CmdHelp() 
  15. }  

 

在‘DockerCli.getMethod’中,我们可以看到它是通过对一个函数的动态调用实现的,其中这个函数名的形式为在Docker命令前预置 “Cmd”字符串。那么在‘docker build’这个情况下,我们寻找的是‘DockerCli.CmdBuild’。但在这个文件中并没有对应的方法,因此让我们需要搜索‘CmdBuild’。

 

  1. func (cli *DockerCli) getMethod(args ...string) (func(...string) error, bool) { 
  2. camelArgs := make([]string, len(args)) 
  3. for i, s := range args { 
  4.     if len(s) == 0 { 
  5.         return nil, false 
  6.     } 
  7.     camelArgs[i] = strings.ToUpper(s[:1]) + strings.ToLower(s[1:]) 
  8. methodName := "Cmd" + strings.Join(camelArgs, ""
  9. method := reflect.ValueOf(cli).MethodByName(methodName) 
  10. if !method.IsValid() { 
  11.     return nil, false 
  12. return method.Interface().(func(...string) error), true 
  13. }  

 

搜索结果显示‘DockerCli’中确实有一个‘CmdBuild’方法,因此跳到它的定义部分。由于‘DockerCli.CmdBuild’的方法体过长,因此就不在本文中嵌入了,但是这里有它的链接。

这里有很多内容。在方法的顶部,我们可以看到代码会为Dockerfile和配置处理各种输入方法。通常,在阅读一个很长的方法时,倒过来读是一种很不错的策略。从底部开始,观察函数在***做了什么。很多情况中,它们都是函数的本质,而之前的内容无非只是用来补全核心行为的。

在‘CmdBuild’的底部,我们可以看到通过‘cli.stream’构造的‘POST’请求。通过一些额外定义的跳转,我们到达了‘DockerCli.clientRequest’,它构造一个HTTP请求,这个请求包含你通过‘docker build’传递给Docker的信息。因此在这里,‘docker build所做的就是发出一个设想的’POST‘请求给Docker守护进程。如果你愿意,你也可以使用’curl‘来完成这个行为。

至此,我们已经彻底了解了一个单独的Docker客户端命令,或许你仍希望更进一步,找到守护进程接受请求的部分,并一路跟踪到它和LXC以及内核交互的部分。这当然是一条合理的路径,但是我们将其作为练习留给各位读者。接下来,让我们对客户端的关键组件有一个更加全面的认识。

#p#

步骤4:查看使用示例

更好地理解一段代码的方式是查看展示代码如何被应用的使用示例。让我们回到'DockerCli.clientRequest'方法。在右手边的Sourcegraph面板中,我们可以浏览这个方法的使用例子。结果显示,这个方法在多处被使用,因为大部分Docker客户端命令都会产生传到守护进程的HTTP请求。

 

五步助你成为优秀的Docker代码贡献者

为了完全理解一个代码片段,你需要同时知晓它是如何工作的以及是如何来使用的。通过阅读代码的定义部分让我们理解前者,而查看使用示例则是涵盖了后者。

请在更多的函数和方法上尝试,理解它们的内部联系。如果这有帮助,那么请就应用的不同模块如何交互,画一张图。

步骤5:选择一个问题并开始coding

既然你已经对Docker的代码基有了一个大概的认识,那么可以查阅一下issue跟踪系统,看看哪些问题亟待解决,并在遇到你自己无法回答的问题时,向Docker社区的成员申援。由于你已经花了时间来摸索并理解代码,那么你应该已经具备条件来提出“聪明”的问题,并知道问题大概出在哪里。

如果你觉得有必要,可以一路做好笔记,记录你的经历,并像本文一样作为博客发布。Docker团队会很乐意看到,你研究他们代码的经历。

有效地贡献

对一个巨大且陌生的基准代码的恐惧,俨然已经成为了一个阻止人们参与到项目中的误解。我们经常假设,对于程序员而言,工作的难点在于写代码,然而阅读并理解他人的代码却往往是最关键的一步。认识到这一切,并坚定地迎接任务,辅以优秀的工具,会帮助你克服心理防线,以更好地投入到代码中。

那么,开始动手吧,检查一下Docker今天的代码。一个充满活力的开源社区和基准代码正等着你!

原文链接:http://dockone.io/article/450

责任编辑:Ophira 来源: dockerone
相关推荐

2019-12-18 23:11:24

TF架构网络连接

2015-07-22 16:08:46

OpenStack开源贡献代码

2020-06-18 11:14:53

微软谷歌开源

2015-05-19 09:11:32

OpenStackOpenStack贡献

2020-04-17 13:01:38

ASFApache董事会

2022-03-26 10:18:26

GoogleRust获奖者

2009-07-21 08:41:52

Linux内核开源操作系统Intel

2016-02-01 09:24:24

Quora排行算法

2015-09-08 09:05:16

贡献者维基Linux

2013-09-09 12:35:54

MongoDB

2019-01-21 08:00:00

谷歌开源数据

2011-07-01 09:26:12

2023-09-19 07:20:33

2016-10-27 16:03:28

Easystack开源

2021-07-07 09:41:16

CentOS CentOS StreCentOS Stre

2024-02-04 13:33:57

2022-06-08 08:55:15

JavaScript代码前端

2012-11-13 10:47:59

大数据HBaseHadoop

2021-09-17 16:05:06

Google开源贡献者获奖者
点赞
收藏

51CTO技术栈公众号