Golang应用程序自动构建版本控制

时间:2012-07-06 00:52:01

标签: go

每次编译Go应用程序时,是否可以自动增加次要版本号?

我想在我的程序中设置一个版本号,带有一个自动增量部分:

$ myapp -version
MyApp version 0.5.132

我设置的版本号是0.5,而132是每次编译二进制文件时自动递增的值。

这可以在Go吗?

6 个答案:

答案 0 :(得分:297)

Go链接器(go tool link)可以选择设置未初始化的字符串变量的值:

-X importpath.name=value
  Set the value of the string variable in importpath named name to

值。         请注意,在Go 1.5之前,此选项采用了两个单独的参数。         现在它需要在第一个=符号上拆分一个参数。

作为构建过程的一部分,您可以使用此设置版本字符串变量。您可以使用go通过-ldflags工具传递此信息。例如,给定以下源文件:

package main

import "fmt"

var xyz string

func main() {
    fmt.Println(xyz)
}

然后:

$ go run -ldflags "-X main.xyz=abc" main.go
abc

为了将main.minversion设置为构建时的构建日期和时间:

go build -ldflags "-X main.minversion=`date -u +.%Y%m%d.%H%M%S`" service.go

如果编译时没有以这种方式初始化main.minversion,它将包含空字符串。

答案 1 :(得分:23)

在构建混合命令行应用程序和库项目时,我无法使用-ldflags参数,因此我最终使用Makefile目标生成包含应用程序版本和构建日期的Go源文件: / p>

BUILD_DATE := `date +%Y-%m-%d\ %H:%M`
VERSIONFILE := cmd/myapp/version.go

gensrc:
    rm -f $(VERSIONFILE)
    @echo "package main" > $(VERSIONFILE)
    @echo "const (" >> $(VERSIONFILE)
    @echo "  VERSION = \"1.0\"" >> $(VERSIONFILE)
    @echo "  BUILD_DATE = \"$(BUILD_DATE)\"" >> $(VERSIONFILE)
    @echo ")" >> $(VERSIONFILE)

在我的init()方法中,我这样做:

flag.Usage = func() {
    fmt.Fprintf(os.Stderr, "%s version %s\n", os.Args[0], VERSION)
    fmt.Fprintf(os.Stderr, "built %s\n", BUILD_DATE)
    fmt.Fprintln(os.Stderr, "usage:")
    flag.PrintDefaults()
}

但是,如果您想要原子增加的内部版本号而不是构建日期,则可能需要创建一个包含最后一个内部版本号的本地文件。您的Makefile会将文件内容读入变量,递增,将其插入version.go文件而不是日期,然后将新的内部版本编号写回文件。

答案 2 :(得分:17)

此外,我想发布一个小例子如何使用git和makefile:

--- Makefile ----

# This how we want to name the binary output
BINARY=gomake

# These are the values we want to pass for VERSION and BUILD
# git tag 1.0.1
# git commit -am "One more change after the tags"
VERSION=`git describe --tags`
BUILD=`date +%FT%T%z`

# Setup the -ldflags option for go build here, interpolate the variable values
LDFLAGS_f1=-ldflags "-w -s -X main.Version=${VERSION} -X main.Build=${BUILD} -X main.Entry=f1"
LDFLAGS_f2=-ldflags "-w -s -X main.Version=${VERSION} -X main.Build=${BUILD} -X main.Entry=f2"

# Builds the project
build:
    go build ${LDFLAGS_f1} -o ${BINARY}_f1
    go build ${LDFLAGS_f2} -o ${BINARY}_f2

# Installs our project: copies binaries
install:
    go install ${LDFLAGS_f1}

# Cleans our project: deletes binaries
clean:
    if [ -f ${BINARY} ] ; then rm ${BINARY} ; fi

.PHONY: clean install

make文件将创建两个可执行文件。一个是执行功能一,另一个是将功能二作为主要条目:

package main

import (
        "fmt"
)

var (

        Version string
        Build   string
        Entry   string

        funcs = map[string]func() {
                "f1":functionOne,"f2":functionTwo,
        }

)

func functionOne() {
    fmt.Println("This is function one")
}

func functionTwo() {
    fmt.Println("This is function two")
}

func main() {

        fmt.Println("Version: ", Version)
        fmt.Println("Build Time: ", Build)

    funcs[Entry]()

}

然后运行:

make

你会得到:

mab@h2470988:~/projects/go/gomake/3/gomake$ ls -al
total 2020
drwxrwxr-x 3 mab mab    4096 Sep  7 22:41 .
drwxrwxr-x 3 mab mab    4096 Aug 16 10:00 ..
drwxrwxr-x 8 mab mab    4096 Aug 17 16:40 .git
-rwxrwxr-x 1 mab mab 1023488 Sep  7 22:41 gomake_f1
-rwxrwxr-x 1 mab mab 1023488 Sep  7 22:41 gomake_f2
-rw-rw-r-- 1 mab mab     399 Aug 16 10:21 main.go
-rw-rw-r-- 1 mab mab     810 Sep  7 22:41 Makefile
mab@h2470988:~/projects/go/gomake/3/gomake$ ./gomake_f1
Version:  1.0.1-1-gfb51187
Build Time:  2016-09-07T22:41:38+0200
This is function one
mab@h2470988:~/projects/go/gomake/3/gomake$ ./gomake_f2
Version:  1.0.1-1-gfb51187
Build Time:  2016-09-07T22:41:39+0200
This is function two

答案 3 :(得分:11)

使用多-ldflags

$ go build -ldflags "-X name1=value1 -X name2=value2" -o path/to/output

答案 4 :(得分:10)

在Windows操作系统上给出以下程序

package main

import "fmt"

var (
    version string
    date    string
)

func main() {
    fmt.Printf("version=%s, date=%s", version, date)
}

您可以使用

进行构建
go build -ldflags "-X main.version=0.0.1 -X main.date=%date:~10,4%-%date:~4,2%-%date:~7,2%T%time:~0,2%:%time:~3,2%:%time:~6,2%"

日期格式假设您的环境echo %date%Fri 07/22/2016echo %time%16:21:52.88

然后输出将是:version=0.0.1, date=2016-07-22T16:21:52

答案 5 :(得分:3)

使用ldflags来设置main程序包中的变量:

带有文件main.go

package main

import "fmt"

var (
    version string
    build   string
)

func main() {
    fmt.Println("version=", version)
    fmt.Println("build=", build)
}

然后运行:

go run \
  -ldflags "-X main.version=1.0.0 -X main.build=12082019" \ 
  main.go

内部版本:

go build -o mybinary \
  -ldflags "-X main.version=1.0.0 -X 'main.build=$(date)'" \ 
  main.go

使用ldflags在非main包中设置变量:

带有文件config.go

package config

import "fmt"

var (
    Version string
)

func LogVersion() {
    fmt.Println("version=", Version)
}

您还将需要文件main.go

package main

import (
    "fmt"
    "github.com/user/repo/config"
}

func main() {
    config.LogVersion()
}

首先构建您的二进制文件:

go build -o mybinary main.go 

找到要设置的变量名的完整路径:

go tool nm <path_to_binary> | grep Version

再次运行并构建二进制文件,但要使用ldflags

go run \
  -ldflags "-X github.com/user/repo/config.Version=1.0.0" \
  main.go --version       


go build -o mybinary \
  -ldflags "-X github.com/user/repo/config.Version=1.0.0" \
  main.go     

受到https://github.com/golang/go/wiki/GcToolchainTricks#including-build-information-in-the-executable

的启发

如果您使用的是goreleaser,请阅读此https://goreleaser.com/#using-the-main-version

  

默认明智的GoReleaser设置三个ldflag:

     

main.version:当前的Git标签
  main.commit:当前git commit SHA
  main.date:日期符合RFC3339