2016-05-25 14:07:57 +02:00
// Copyright 2016 The go-ethereum Authors
// This file is part of the go-ethereum library.
//
// The go-ethereum library is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// The go-ethereum library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
// +build none
/ *
The ci command is called from Continuous Integration scripts .
Usage : go run ci . go < command > < command flags / arguments >
Available commands are :
2017-03-01 10:47:55 +02:00
install [ - arch architecture ] [ packages ... ] -- builds packages and executables
test [ - coverage ] [ - vet ] [ - misspell ] [ packages ... ] -- runs the tests
archive [ - arch architecture ] [ - type zip | tar ] [ - signer key - envvar ] [ - upload dest ] -- archives build artefacts
importkeys -- imports signing keys from env
debsrc [ - signer key - id ] [ - upload dest ] -- creates a debian source package
nsis -- creates a Windows NSIS installer
aar [ - local ] [ - sign key - id ] [ - deploy repo ] [ - upload dest ] -- creates an Android archive
xcode [ - local ] [ - sign key - id ] [ - deploy repo ] [ - upload dest ] -- creates an iOS XCode framework
xgo [ - alltools ] [ options ] -- cross builds according to options
2016-05-25 14:07:57 +02:00
For all commands , - n prevents execution of external programs ( dry run mode ) .
* /
package main
import (
2016-10-17 15:17:14 +03:00
"bufio"
2016-05-25 14:07:57 +02:00
"bytes"
"encoding/base64"
"flag"
"fmt"
2016-11-02 19:36:48 +02:00
"go/parser"
"go/token"
2016-05-25 14:07:57 +02:00
"io/ioutil"
"log"
"os"
"os/exec"
"path/filepath"
2016-10-17 15:17:14 +03:00
"regexp"
2016-05-25 14:07:57 +02:00
"runtime"
"strings"
"time"
2016-11-02 18:21:05 +02:00
"github.com/ethereum/go-ethereum/internal/build"
2016-05-25 14:07:57 +02:00
)
var (
// Files that end up in the geth*.zip archive.
gethArchiveFiles = [ ] string {
"COPYING" ,
executablePath ( "geth" ) ,
}
// Files that end up in the geth-alltools*.zip archive.
allToolsArchiveFiles = [ ] string {
"COPYING" ,
executablePath ( "abigen" ) ,
2017-03-02 15:23:15 +02:00
executablePath ( "bootnode" ) ,
2016-05-25 14:07:57 +02:00
executablePath ( "evm" ) ,
executablePath ( "geth" ) ,
2016-12-13 09:18:18 +01:00
executablePath ( "swarm" ) ,
2016-05-25 14:07:57 +02:00
executablePath ( "rlpdump" ) ,
}
// A debian package is created for all executables listed here.
debExecutables = [ ] debExecutable {
{
Name : "geth" ,
Description : "Ethereum CLI client." ,
} ,
2017-03-09 12:02:43 +02:00
{
Name : "bootnode" ,
Description : "Ethereum bootnode." ,
} ,
2016-05-25 14:07:57 +02:00
{
Name : "rlpdump" ,
Description : "Developer utility tool that prints RLP structures." ,
} ,
{
Name : "evm" ,
Description : "Developer utility version of the EVM (Ethereum Virtual Machine) that is capable of running bytecode snippets within a configurable environment and execution mode." ,
} ,
2016-11-28 11:25:19 +01:00
{
2016-12-13 09:18:18 +01:00
Name : "swarm" ,
Description : "Ethereum Swarm daemon and tools" ,
2016-11-28 11:25:19 +01:00
} ,
2016-05-25 14:07:57 +02:00
{
Name : "abigen" ,
Description : "Source code generator to convert Ethereum contract definitions into easy to use, compile-time type-safe Go packages." ,
} ,
}
// Distros for which packages are created.
// Note: vivid is unsupported because there is no golang-1.6 package for it.
2016-12-14 11:45:14 +02:00
// Note: wily is unsupported because it was officially deprecated on lanchpad.
debDistros = [ ] string { "trusty" , "xenial" , "yakkety" }
2016-05-25 14:07:57 +02:00
)
var GOBIN , _ = filepath . Abs ( filepath . Join ( "build" , "bin" ) )
func executablePath ( name string ) string {
if runtime . GOOS == "windows" {
name += ".exe"
}
return filepath . Join ( GOBIN , name )
}
func main ( ) {
log . SetFlags ( log . Lshortfile )
if _ , err := os . Stat ( filepath . Join ( "build" , "ci.go" ) ) ; os . IsNotExist ( err ) {
log . Fatal ( "this script must be run from the root of the repository" )
}
if len ( os . Args ) < 2 {
log . Fatal ( "need subcommand as first argument" )
}
switch os . Args [ 1 ] {
case "install" :
doInstall ( os . Args [ 2 : ] )
case "test" :
doTest ( os . Args [ 2 : ] )
case "archive" :
doArchive ( os . Args [ 2 : ] )
case "debsrc" :
doDebianSource ( os . Args [ 2 : ] )
2016-11-08 23:55:39 +02:00
case "nsis" :
doWindowsInstaller ( os . Args [ 2 : ] )
2016-10-17 15:17:14 +03:00
case "aar" :
doAndroidArchive ( os . Args [ 2 : ] )
2016-11-09 11:13:37 +02:00
case "xcode" :
doXCodeFramework ( os . Args [ 2 : ] )
2016-08-08 13:41:55 +03:00
case "xgo" :
doXgo ( os . Args [ 2 : ] )
2016-05-25 14:07:57 +02:00
default :
log . Fatal ( "unknown command " , os . Args [ 1 ] )
}
}
// Compiling
func doInstall ( cmdline [ ] string ) {
2016-11-02 19:36:48 +02:00
var (
arch = flag . String ( "arch" , "" , "Architecture to cross build for" )
)
2016-05-25 14:07:57 +02:00
flag . CommandLine . Parse ( cmdline )
2016-09-30 22:03:08 +02:00
env := build . Env ( )
2016-05-25 14:07:57 +02:00
// Check Go version. People regularly open issues about compilation
// failure with outdated Go. This should save them the trouble.
2017-03-22 15:52:44 +01:00
if runtime . Version ( ) < "go1.7" && ! strings . HasPrefix ( runtime . Version ( ) , "devel" ) {
2016-05-25 14:07:57 +02:00
log . Println ( "You have Go version" , runtime . Version ( ) )
2017-03-22 15:52:44 +01:00
log . Println ( "go-ethereum requires at least Go version 1.7 and cannot" )
2016-05-25 14:07:57 +02:00
log . Println ( "be compiled with an earlier version. Please upgrade your Go installation." )
os . Exit ( 1 )
}
// Compile packages given as arguments, or everything if there are no arguments.
packages := [ ] string { "./..." }
if flag . NArg ( ) > 0 {
packages = flag . Args ( )
}
2016-11-02 19:36:48 +02:00
if * arch == "" || * arch == runtime . GOARCH {
goinstall := goTool ( "install" , buildFlags ( env ) ... )
goinstall . Args = append ( goinstall . Args , "-v" )
goinstall . Args = append ( goinstall . Args , packages ... )
build . MustRun ( goinstall )
return
}
2016-11-04 13:48:20 +02:00
// If we are cross compiling to ARMv5 ARMv6 or ARMv7, clean any prvious builds
if * arch == "arm" {
os . RemoveAll ( filepath . Join ( runtime . GOROOT ( ) , "pkg" , runtime . GOOS + "_arm" ) )
for _ , path := range filepath . SplitList ( build . GOPATH ( ) ) {
os . RemoveAll ( filepath . Join ( path , "pkg" , runtime . GOOS + "_arm" ) )
}
}
2016-11-02 19:36:48 +02:00
// Seems we are cross compiling, work around forbidden GOBIN
goinstall := goToolArch ( * arch , "install" , buildFlags ( env ) ... )
2016-05-25 14:07:57 +02:00
goinstall . Args = append ( goinstall . Args , "-v" )
2016-11-02 19:36:48 +02:00
goinstall . Args = append ( goinstall . Args , [ ] string { "-buildmode" , "archive" } ... )
2016-05-25 14:07:57 +02:00
goinstall . Args = append ( goinstall . Args , packages ... )
build . MustRun ( goinstall )
2016-11-02 19:36:48 +02:00
if cmds , err := ioutil . ReadDir ( "cmd" ) ; err == nil {
for _ , cmd := range cmds {
pkgs , err := parser . ParseDir ( token . NewFileSet ( ) , filepath . Join ( "." , "cmd" , cmd . Name ( ) ) , nil , parser . PackageClauseOnly )
if err != nil {
log . Fatal ( err )
}
2017-01-06 15:52:03 +01:00
for name := range pkgs {
2016-11-02 19:36:48 +02:00
if name == "main" {
gobuild := goToolArch ( * arch , "build" , buildFlags ( env ) ... )
gobuild . Args = append ( gobuild . Args , "-v" )
2016-11-07 12:18:28 +02:00
gobuild . Args = append ( gobuild . Args , [ ] string { "-o" , executablePath ( cmd . Name ( ) ) } ... )
2016-11-02 19:36:48 +02:00
gobuild . Args = append ( gobuild . Args , "." + string ( filepath . Separator ) + filepath . Join ( "cmd" , cmd . Name ( ) ) )
build . MustRun ( gobuild )
break
}
}
}
}
2016-05-25 14:07:57 +02:00
}
2016-09-30 22:03:08 +02:00
func buildFlags ( env build . Environment ) ( flags [ ] string ) {
// Set gitCommit constant via link-time assignment.
if env . Commit != "" {
2017-03-22 15:52:44 +01:00
flags = append ( flags , "-ldflags" , "-X main.gitCommit=" + env . Commit )
2016-05-25 14:07:57 +02:00
}
return flags
}
func goTool ( subcmd string , args ... string ) * exec . Cmd {
2016-11-02 19:36:48 +02:00
return goToolArch ( runtime . GOARCH , subcmd , args ... )
}
func goToolArch ( arch string , subcmd string , args ... string ) * exec . Cmd {
2016-05-25 14:07:57 +02:00
gocmd := filepath . Join ( runtime . GOROOT ( ) , "bin" , "go" )
cmd := exec . Command ( gocmd , subcmd )
cmd . Args = append ( cmd . Args , args ... )
2017-01-26 16:41:32 +02:00
if subcmd == "build" || subcmd == "install" || subcmd == "test" {
// Go CGO has a Windows linker error prior to 1.8 (https://github.com/golang/go/issues/8756).
// Work around issue by allowing multiple definitions for <1.8 builds.
if runtime . GOOS == "windows" && runtime . Version ( ) < "go1.8" {
cmd . Args = append ( cmd . Args , [ ] string { "-ldflags" , "-extldflags -Wl,--allow-multiple-definition" } ... )
}
}
2017-03-22 15:52:44 +01:00
cmd . Env = [ ] string { "GOPATH=" + build . GOPATH ( ) }
2016-11-02 19:36:48 +02:00
if arch == "" || arch == runtime . GOARCH {
cmd . Env = append ( cmd . Env , "GOBIN=" + GOBIN )
} else {
cmd . Env = append ( cmd . Env , "CGO_ENABLED=1" )
cmd . Env = append ( cmd . Env , "GOARCH=" + arch )
2016-05-25 14:07:57 +02:00
}
for _ , e := range os . Environ ( ) {
if strings . HasPrefix ( e , "GOPATH=" ) || strings . HasPrefix ( e , "GOBIN=" ) {
continue
}
cmd . Env = append ( cmd . Env , e )
}
return cmd
}
// Running The Tests
//
// "tests" also includes static analysis tools such as vet.
func doTest ( cmdline [ ] string ) {
var (
vet = flag . Bool ( "vet" , false , "Whether to run go vet" )
2017-01-13 10:53:43 +02:00
misspell = flag . Bool ( "misspell" , false , "Whether to run the spell checker" )
2016-05-25 14:07:57 +02:00
coverage = flag . Bool ( "coverage" , false , "Whether to record code coverage" )
)
flag . CommandLine . Parse ( cmdline )
2016-10-28 20:05:01 +03:00
2016-05-25 14:07:57 +02:00
packages := [ ] string { "./..." }
if len ( flag . CommandLine . Args ( ) ) > 0 {
packages = flag . CommandLine . Args ( )
}
2016-10-28 20:05:01 +03:00
if len ( packages ) == 1 && packages [ 0 ] == "./..." {
// Resolve ./... manually since go vet will fail on vendored stuff
out , err := goTool ( "list" , "./..." ) . CombinedOutput ( )
if err != nil {
log . Fatalf ( "package listing failed: %v\n%s" , err , string ( out ) )
}
packages = [ ] string { }
for _ , line := range strings . Split ( string ( out ) , "\n" ) {
if ! strings . Contains ( line , "vendor" ) {
packages = append ( packages , strings . TrimSpace ( line ) )
}
}
}
2016-05-25 14:07:57 +02:00
// Run analysis tools before the tests.
if * vet {
build . MustRun ( goTool ( "vet" , packages ... ) )
}
2017-01-13 10:53:43 +02:00
if * misspell {
2017-01-13 11:54:17 +02:00
spellcheck ( packages )
2017-01-13 10:53:43 +02:00
}
2016-05-25 14:07:57 +02:00
// Run the actual tests.
gotest := goTool ( "test" )
2016-09-26 13:41:18 +02:00
// Test a single package at a time. CI builders are slow
// and some tests run into timeouts under load.
gotest . Args = append ( gotest . Args , "-p" , "1" )
2016-05-25 14:07:57 +02:00
if * coverage {
gotest . Args = append ( gotest . Args , "-covermode=atomic" , "-cover" )
}
gotest . Args = append ( gotest . Args , packages ... )
build . MustRun ( gotest )
}
2017-01-13 11:54:17 +02:00
// spellcheck runs the client9/misspell spellchecker package on all Go, Cgo and
// test files in the requested packages.
func spellcheck ( packages [ ] string ) {
// Ensure the spellchecker is available
build . MustRun ( goTool ( "get" , "github.com/client9/misspell/cmd/misspell" ) )
2017-01-30 00:55:41 +09:00
// Windows chokes on long argument lists, check packages individually
2017-01-13 11:54:17 +02:00
for _ , pkg := range packages {
// The spell checker doesn't work on packages, gather all .go files for it
out , err := goTool ( "list" , "-f" , "{{.Dir}}{{range .GoFiles}}\n{{.}}{{end}}{{range .CgoFiles}}\n{{.}}{{end}}{{range .TestGoFiles}}\n{{.}}{{end}}" , pkg ) . CombinedOutput ( )
if err != nil {
log . Fatalf ( "source file listing failed: %v\n%s" , err , string ( out ) )
}
// Retrieve the folder and assemble the source list
lines := strings . Split ( string ( out ) , "\n" )
root := lines [ 0 ]
sources := make ( [ ] string , 0 , len ( lines ) - 1 )
for _ , line := range lines [ 1 : ] {
if line = strings . TrimSpace ( line ) ; line != "" {
sources = append ( sources , filepath . Join ( root , line ) )
}
}
// Run the spell checker for this particular package
build . MustRunCommand ( filepath . Join ( GOBIN , "misspell" ) , append ( [ ] string { "-error" } , sources ... ) ... )
}
}
2016-05-25 14:07:57 +02:00
// Release Packaging
func doArchive ( cmdline [ ] string ) {
var (
2016-11-02 19:36:48 +02:00
arch = flag . String ( "arch" , runtime . GOARCH , "Architecture cross packaging" )
2016-11-02 18:21:05 +02:00
atype = flag . String ( "type" , "zip" , "Type of archive to write (zip|tar)" )
signer = flag . String ( "signer" , "" , ` Environment variable holding the signing key (e.g. LINUX_SIGNING_KEY) ` )
upload = flag . String ( "upload" , "" , ` Destination to upload the archives (usually "gethstore/builds") ` )
ext string
2016-05-25 14:07:57 +02:00
)
flag . CommandLine . Parse ( cmdline )
switch * atype {
case "zip" :
ext = ".zip"
case "tar" :
ext = ".tar.gz"
default :
log . Fatal ( "unknown archive type: " , atype )
}
2016-09-30 22:03:08 +02:00
2016-11-03 13:44:16 +01:00
var (
env = build . Env ( )
base = archiveBasename ( * arch , env )
geth = "geth-" + base + ext
alltools = "geth-alltools-" + base + ext
)
2016-09-30 22:03:08 +02:00
maybeSkipArchive ( env )
2016-11-03 13:44:16 +01:00
if err := build . WriteArchive ( geth , gethArchiveFiles ) ; err != nil {
2016-05-25 14:07:57 +02:00
log . Fatal ( err )
}
2016-11-03 13:44:16 +01:00
if err := build . WriteArchive ( alltools , allToolsArchiveFiles ) ; err != nil {
2016-05-25 14:07:57 +02:00
log . Fatal ( err )
}
2016-11-03 13:44:16 +01:00
for _ , archive := range [ ] string { geth , alltools } {
2016-11-02 18:21:05 +02:00
if err := archiveUpload ( archive , * upload , * signer ) ; err != nil {
log . Fatal ( err )
}
}
2016-05-25 14:07:57 +02:00
}
2016-11-02 19:36:48 +02:00
func archiveBasename ( arch string , env build . Environment ) string {
platform := runtime . GOOS + "-" + arch
2016-11-04 13:48:20 +02:00
if arch == "arm" {
platform += os . Getenv ( "GOARM" )
}
2016-10-17 15:17:14 +03:00
if arch == "android" {
platform = "android-all"
}
2016-11-09 11:13:37 +02:00
if arch == "ios" {
platform = "ios-all"
}
return platform + "-" + archiveVersion ( env )
}
func archiveVersion ( env build . Environment ) string {
version := build . VERSION ( )
2016-11-03 13:44:16 +01:00
if isUnstableBuild ( env ) {
2016-11-09 11:13:37 +02:00
version += "-unstable"
2016-11-03 13:44:16 +01:00
}
2016-09-30 22:03:08 +02:00
if env . Commit != "" {
2016-11-09 11:13:37 +02:00
version += "-" + env . Commit [ : 8 ]
2016-05-25 14:07:57 +02:00
}
2016-11-09 11:13:37 +02:00
return version
2016-05-25 14:07:57 +02:00
}
2016-11-02 18:21:05 +02:00
func archiveUpload ( archive string , blobstore string , signer string ) error {
// If signing was requested, generate the signature files
if signer != "" {
pgpkey , err := base64 . StdEncoding . DecodeString ( os . Getenv ( signer ) )
if err != nil {
return fmt . Errorf ( "invalid base64 %s" , signer )
}
if err := build . PGPSignFile ( archive , archive + ".asc" , string ( pgpkey ) ) ; err != nil {
return err
}
}
// If uploading to Azure was requested, push the archive possibly with its signature
if blobstore != "" {
auth := build . AzureBlobstoreConfig {
Account : strings . Split ( blobstore , "/" ) [ 0 ] ,
Token : os . Getenv ( "AZURE_BLOBSTORE_TOKEN" ) ,
Container : strings . SplitN ( blobstore , "/" , 2 ) [ 1 ] ,
}
2016-11-08 23:46:46 +01:00
if err := build . AzureBlobstoreUpload ( archive , filepath . Base ( archive ) , auth ) ; err != nil {
2016-11-02 18:21:05 +02:00
return err
}
if signer != "" {
2016-11-08 23:46:46 +01:00
if err := build . AzureBlobstoreUpload ( archive + ".asc" , filepath . Base ( archive + ".asc" ) , auth ) ; err != nil {
2016-11-02 18:21:05 +02:00
return err
}
}
}
return nil
}
2016-09-30 22:03:08 +02:00
// skips archiving for some build configurations.
func maybeSkipArchive ( env build . Environment ) {
if env . IsPullRequest {
log . Printf ( "skipping because this is a PR build" )
os . Exit ( 0 )
}
2016-11-15 14:11:27 +01:00
if env . Branch != "master" && ! strings . HasPrefix ( env . Tag , "v1." ) {
2016-09-30 22:03:08 +02:00
log . Printf ( "skipping because branch %q, tag %q is not on the whitelist" , env . Branch , env . Tag )
os . Exit ( 0 )
}
}
2016-05-25 14:07:57 +02:00
// Debian Packaging
2016-09-30 22:03:08 +02:00
func doDebianSource ( cmdline [ ] string ) {
var (
signer = flag . String ( "signer" , "" , ` Signing key name, also used as package author ` )
upload = flag . String ( "upload" , "" , ` Where to upload the source package (usually "ppa:ethereum/ethereum") ` )
workdir = flag . String ( "workdir" , "" , ` Output directory for packages (uses temp dir if unset) ` )
now = time . Now ( )
)
2016-05-25 14:07:57 +02:00
flag . CommandLine . Parse ( cmdline )
2016-09-30 22:03:08 +02:00
* workdir = makeWorkdir ( * workdir )
env := build . Env ( )
maybeSkipArchive ( env )
2016-05-25 14:07:57 +02:00
// Import the signing key.
if b64key := os . Getenv ( "PPA_SIGNING_KEY" ) ; b64key != "" {
key , err := base64 . StdEncoding . DecodeString ( b64key )
if err != nil {
log . Fatal ( "invalid base64 PPA_SIGNING_KEY" )
}
gpg := exec . Command ( "gpg" , "--import" )
gpg . Stdin = bytes . NewReader ( key )
build . MustRun ( gpg )
}
2016-09-30 22:03:08 +02:00
// Create the packages.
2016-05-25 14:07:57 +02:00
for _ , distro := range debDistros {
2016-09-30 22:03:08 +02:00
meta := newDebMetadata ( distro , * signer , env , now )
pkgdir := stageDebianSource ( * workdir , meta )
2016-05-25 14:07:57 +02:00
debuild := exec . Command ( "debuild" , "-S" , "-sa" , "-us" , "-uc" )
debuild . Dir = pkgdir
build . MustRun ( debuild )
changes := fmt . Sprintf ( "%s_%s_source.changes" , meta . Name ( ) , meta . VersionString ( ) )
2016-09-30 22:03:08 +02:00
changes = filepath . Join ( * workdir , changes )
2016-05-25 14:07:57 +02:00
if * signer != "" {
build . MustRunCommand ( "debsign" , changes )
}
if * upload != "" {
build . MustRunCommand ( "dput" , * upload , changes )
}
}
}
2016-09-30 22:03:08 +02:00
func makeWorkdir ( wdflag string ) string {
var err error
if wdflag != "" {
err = os . MkdirAll ( wdflag , 0744 )
} else {
2016-11-08 23:55:39 +02:00
wdflag , err = ioutil . TempDir ( "" , "geth-build-" )
2016-09-30 22:03:08 +02:00
}
if err != nil {
log . Fatal ( err )
}
return wdflag
}
func isUnstableBuild ( env build . Environment ) bool {
2016-11-18 19:55:19 +01:00
if env . Tag != "" {
2016-09-30 22:03:08 +02:00
return false
}
return true
2016-05-25 14:07:57 +02:00
}
type debMetadata struct {
2016-09-30 22:03:08 +02:00
Env build . Environment
2016-05-25 14:07:57 +02:00
// go-ethereum version being built. Note that this
// is not the debian package version. The package version
// is constructed by VersionString.
Version string
2016-09-30 22:03:08 +02:00
Author string // "name <email>", also selects signing key
Distro , Time string
Executables [ ] debExecutable
2016-05-25 14:07:57 +02:00
}
2016-09-30 22:03:08 +02:00
type debExecutable struct {
Name , Description string
}
func newDebMetadata ( distro , author string , env build . Environment , t time . Time ) debMetadata {
2016-05-25 14:07:57 +02:00
if author == "" {
// No signing key, use default author.
author = "Ethereum Builds <fjl@ethereum.org>"
}
return debMetadata {
2016-09-30 22:03:08 +02:00
Env : env ,
2016-05-25 14:07:57 +02:00
Author : author ,
Distro : distro ,
Version : build . VERSION ( ) ,
Time : t . Format ( time . RFC1123Z ) ,
Executables : debExecutables ,
}
}
// Name returns the name of the metapackage that depends
// on all executable packages.
func ( meta debMetadata ) Name ( ) string {
2016-09-30 22:03:08 +02:00
if isUnstableBuild ( meta . Env ) {
2016-05-25 14:07:57 +02:00
return "ethereum-unstable"
}
return "ethereum"
}
// VersionString returns the debian version of the packages.
func ( meta debMetadata ) VersionString ( ) string {
vsn := meta . Version
2016-09-30 22:03:08 +02:00
if meta . Env . Buildnum != "" {
vsn += "+build" + meta . Env . Buildnum
2016-05-25 14:07:57 +02:00
}
if meta . Distro != "" {
vsn += "+" + meta . Distro
}
return vsn
}
// ExeList returns the list of all executable packages.
func ( meta debMetadata ) ExeList ( ) string {
names := make ( [ ] string , len ( meta . Executables ) )
for i , e := range meta . Executables {
names [ i ] = meta . ExeName ( e )
}
return strings . Join ( names , ", " )
}
// ExeName returns the package name of an executable package.
func ( meta debMetadata ) ExeName ( exe debExecutable ) string {
2016-09-30 22:03:08 +02:00
if isUnstableBuild ( meta . Env ) {
2016-05-25 14:07:57 +02:00
return exe . Name + "-unstable"
}
return exe . Name
}
// ExeConflicts returns the content of the Conflicts field
// for executable packages.
func ( meta debMetadata ) ExeConflicts ( exe debExecutable ) string {
2016-09-30 22:03:08 +02:00
if isUnstableBuild ( meta . Env ) {
2016-05-25 14:07:57 +02:00
// Set up the conflicts list so that the *-unstable packages
// cannot be installed alongside the regular version.
//
// https://www.debian.org/doc/debian-policy/ch-relationships.html
// is very explicit about Conflicts: and says that Breaks: should
// be preferred and the conflicting files should be handled via
// alternates. We might do this eventually but using a conflict is
// easier now.
return "ethereum, " + exe . Name
}
return ""
}
func stageDebianSource ( tmpdir string , meta debMetadata ) ( pkgdir string ) {
pkg := meta . Name ( ) + "-" + meta . VersionString ( )
pkgdir = filepath . Join ( tmpdir , pkg )
if err := os . Mkdir ( pkgdir , 0755 ) ; err != nil {
log . Fatal ( err )
}
// Copy the source code.
build . MustRunCommand ( "git" , "checkout-index" , "-a" , "--prefix" , pkgdir + string ( filepath . Separator ) )
// Put the debian build files in place.
debian := filepath . Join ( pkgdir , "debian" )
build . Render ( "build/deb.rules" , filepath . Join ( debian , "rules" ) , 0755 , meta )
build . Render ( "build/deb.changelog" , filepath . Join ( debian , "changelog" ) , 0644 , meta )
build . Render ( "build/deb.control" , filepath . Join ( debian , "control" ) , 0644 , meta )
build . Render ( "build/deb.copyright" , filepath . Join ( debian , "copyright" ) , 0644 , meta )
build . RenderString ( "8\n" , filepath . Join ( debian , "compat" ) , 0644 , meta )
build . RenderString ( "3.0 (native)\n" , filepath . Join ( debian , "source/format" ) , 0644 , meta )
for _ , exe := range meta . Executables {
2016-09-30 22:03:08 +02:00
install := filepath . Join ( debian , meta . ExeName ( exe ) + ".install" )
docs := filepath . Join ( debian , meta . ExeName ( exe ) + ".docs" )
2016-05-25 14:07:57 +02:00
build . Render ( "build/deb.install" , install , 0644 , exe )
build . Render ( "build/deb.docs" , docs , 0644 , exe )
}
return pkgdir
}
2016-08-08 13:41:55 +03:00
2016-11-08 23:55:39 +02:00
// Windows installer
func doWindowsInstaller ( cmdline [ ] string ) {
// Parse the flags and make skip installer generation on PRs
var (
arch = flag . String ( "arch" , runtime . GOARCH , "Architecture for cross build packaging" )
signer = flag . String ( "signer" , "" , ` Environment variable holding the signing key (e.g. WINDOWS_SIGNING_KEY) ` )
upload = flag . String ( "upload" , "" , ` Destination to upload the archives (usually "gethstore/builds") ` )
workdir = flag . String ( "workdir" , "" , ` Output directory for packages (uses temp dir if unset) ` )
)
flag . CommandLine . Parse ( cmdline )
* workdir = makeWorkdir ( * workdir )
env := build . Env ( )
maybeSkipArchive ( env )
// Aggregate binaries that are included in the installer
var (
devTools [ ] string
allTools [ ] string
gethTool string
)
for _ , file := range allToolsArchiveFiles {
if file == "COPYING" { // license, copied later
continue
}
allTools = append ( allTools , filepath . Base ( file ) )
if filepath . Base ( file ) == "geth.exe" {
gethTool = file
} else {
devTools = append ( devTools , file )
}
}
// Render NSIS scripts: Installer NSIS contains two installer sections,
// first section contains the geth binary, second section holds the dev tools.
templateData := map [ string ] interface { } {
"License" : "COPYING" ,
"Geth" : gethTool ,
"DevTools" : devTools ,
}
build . Render ( "build/nsis.geth.nsi" , filepath . Join ( * workdir , "geth.nsi" ) , 0644 , nil )
build . Render ( "build/nsis.install.nsh" , filepath . Join ( * workdir , "install.nsh" ) , 0644 , templateData )
build . Render ( "build/nsis.uninstall.nsh" , filepath . Join ( * workdir , "uninstall.nsh" ) , 0644 , allTools )
2016-12-11 00:01:57 +01:00
build . Render ( "build/nsis.pathupdate.nsh" , filepath . Join ( * workdir , "PathUpdate.nsh" ) , 0644 , nil )
2016-11-08 23:55:39 +02:00
build . Render ( "build/nsis.envvarupdate.nsh" , filepath . Join ( * workdir , "EnvVarUpdate.nsh" ) , 0644 , nil )
build . CopyFile ( filepath . Join ( * workdir , "SimpleFC.dll" ) , "build/nsis.simplefc.dll" , 0755 )
build . CopyFile ( filepath . Join ( * workdir , "COPYING" ) , "COPYING" , 0755 )
// Build the installer. This assumes that all the needed files have been previously
// built (don't mix building and packaging to keep cross compilation complexity to a
// minimum).
version := strings . Split ( build . VERSION ( ) , "." )
if env . Commit != "" {
version [ 2 ] += "-" + env . Commit [ : 8 ]
}
installer , _ := filepath . Abs ( "geth-" + archiveBasename ( * arch , env ) + ".exe" )
build . MustRunCommand ( "makensis.exe" ,
"/DOUTPUTFILE=" + installer ,
"/DMAJORVERSION=" + version [ 0 ] ,
"/DMINORVERSION=" + version [ 1 ] ,
"/DBUILDVERSION=" + version [ 2 ] ,
"/DARCH=" + * arch ,
filepath . Join ( * workdir , "geth.nsi" ) ,
)
// Sign and publish installer.
if err := archiveUpload ( installer , * upload , * signer ) ; err != nil {
log . Fatal ( err )
}
2016-11-09 11:13:37 +02:00
}
2016-10-17 15:17:14 +03:00
// Android archives
func doAndroidArchive ( cmdline [ ] string ) {
var (
2016-11-25 12:34:19 +02:00
local = flag . Bool ( "local" , false , ` Flag whether we're only doing a local build (skip Maven artifacts) ` )
2016-10-17 15:17:14 +03:00
signer = flag . String ( "signer" , "" , ` Environment variable holding the signing key (e.g. ANDROID_SIGNING_KEY) ` )
2016-11-09 11:13:37 +02:00
deploy = flag . String ( "deploy" , "" , ` Destination to deploy the archive (usually "https://oss.sonatype.org") ` )
upload = flag . String ( "upload" , "" , ` Destination to upload the archive (usually "gethstore/builds") ` )
2016-10-17 15:17:14 +03:00
)
flag . CommandLine . Parse ( cmdline )
env := build . Env ( )
2017-01-20 11:33:58 +02:00
// Sanity check that the SDK and NDK are installed and set
if os . Getenv ( "ANDROID_HOME" ) == "" {
log . Fatal ( "Please ensure ANDROID_HOME points to your Android SDK" )
}
if os . Getenv ( "ANDROID_NDK" ) == "" {
log . Fatal ( "Please ensure ANDROID_NDK points to your Android NDK" )
}
2016-10-17 15:17:14 +03:00
// Build the Android archive and Maven resources
build . MustRun ( goTool ( "get" , "golang.org/x/mobile/cmd/gomobile" ) )
2017-01-20 11:33:58 +02:00
build . MustRun ( gomobileTool ( "init" , "--ndk" , os . Getenv ( "ANDROID_NDK" ) ) )
2016-10-17 15:17:14 +03:00
build . MustRun ( gomobileTool ( "bind" , "--target" , "android" , "--javapkg" , "org.ethereum" , "-v" , "github.com/ethereum/go-ethereum/mobile" ) )
2016-11-25 12:34:19 +02:00
if * local {
// If we're building locally, copy bundle to build dir and skip Maven
os . Rename ( "geth.aar" , filepath . Join ( GOBIN , "geth.aar" ) )
return
}
2016-10-17 15:17:14 +03:00
meta := newMavenMetadata ( env )
2016-11-09 11:13:37 +02:00
build . Render ( "build/mvn.pom" , meta . Package + ".pom" , 0755 , meta )
2016-10-17 15:17:14 +03:00
// Skip Maven deploy and Azure upload for PR builds
maybeSkipArchive ( env )
2016-11-15 10:25:51 +02:00
// Sign and upload the archive to Azure
archive := "geth-" + archiveBasename ( "android" , env ) + ".aar"
os . Rename ( "geth.aar" , archive )
if err := archiveUpload ( archive , * upload , * signer ) ; err != nil {
log . Fatal ( err )
}
2016-10-17 15:17:14 +03:00
// Sign and upload all the artifacts to Maven Central
2016-11-15 10:25:51 +02:00
os . Rename ( archive , meta . Package + ".aar" )
2016-10-17 15:17:14 +03:00
if * signer != "" && * deploy != "" {
// Import the signing key into the local GPG instance
if b64key := os . Getenv ( * signer ) ; b64key != "" {
key , err := base64 . StdEncoding . DecodeString ( b64key )
if err != nil {
log . Fatalf ( "invalid base64 %s" , * signer )
}
gpg := exec . Command ( "gpg" , "--import" )
gpg . Stdin = bytes . NewReader ( key )
build . MustRun ( gpg )
}
// Upload the artifacts to Sonatype and/or Maven Central
repo := * deploy + "/service/local/staging/deploy/maven2"
2016-11-09 11:13:37 +02:00
if meta . Develop {
2016-10-17 15:17:14 +03:00
repo = * deploy + "/content/repositories/snapshots"
}
build . MustRunCommand ( "mvn" , "gpg:sign-and-deploy-file" ,
"-settings=build/mvn.settings" , "-Durl=" + repo , "-DrepositoryId=ossrh" ,
2016-11-09 11:13:37 +02:00
"-DpomFile=" + meta . Package + ".pom" , "-Dfile=" + meta . Package + ".aar" )
2016-10-17 15:17:14 +03:00
}
}
func gomobileTool ( subcmd string , args ... string ) * exec . Cmd {
cmd := exec . Command ( filepath . Join ( GOBIN , "gomobile" ) , subcmd )
cmd . Args = append ( cmd . Args , args ... )
cmd . Env = [ ] string {
"GOPATH=" + build . GOPATH ( ) ,
}
for _ , e := range os . Environ ( ) {
if strings . HasPrefix ( e , "GOPATH=" ) {
continue
}
cmd . Env = append ( cmd . Env , e )
}
return cmd
}
type mavenMetadata struct {
Version string
2016-11-09 11:13:37 +02:00
Package string
Develop bool
2016-10-17 15:17:14 +03:00
Contributors [ ] mavenContributor
}
type mavenContributor struct {
Name string
Email string
}
func newMavenMetadata ( env build . Environment ) mavenMetadata {
// Collect the list of authors from the repo root
contribs := [ ] mavenContributor { }
if authors , err := os . Open ( "AUTHORS" ) ; err == nil {
defer authors . Close ( )
scanner := bufio . NewScanner ( authors )
for scanner . Scan ( ) {
// Skip any whitespace from the authors list
line := strings . TrimSpace ( scanner . Text ( ) )
if line == "" || line [ 0 ] == '#' {
continue
}
// Split the author and insert as a contributor
2016-11-22 18:35:49 +02:00
re := regexp . MustCompile ( "([^<]+) <(.+)>" )
2016-10-17 15:17:14 +03:00
parts := re . FindStringSubmatch ( line )
if len ( parts ) == 3 {
contribs = append ( contribs , mavenContributor { Name : parts [ 1 ] , Email : parts [ 2 ] } )
}
}
}
2016-11-09 11:13:37 +02:00
// Render the version and package strings
version := build . VERSION ( )
if isUnstableBuild ( env ) {
version += "-SNAPSHOT"
}
2016-10-17 15:17:14 +03:00
return mavenMetadata {
2016-11-09 11:13:37 +02:00
Version : version ,
Package : "geth-" + version ,
Develop : isUnstableBuild ( env ) ,
2016-10-17 15:17:14 +03:00
Contributors : contribs ,
}
}
2016-11-09 11:13:37 +02:00
// XCode frameworks
func doXCodeFramework ( cmdline [ ] string ) {
var (
2016-11-25 12:34:19 +02:00
local = flag . Bool ( "local" , false , ` Flag whether we're only doing a local build (skip Maven artifacts) ` )
2016-11-09 11:13:37 +02:00
signer = flag . String ( "signer" , "" , ` Environment variable holding the signing key (e.g. IOS_SIGNING_KEY) ` )
deploy = flag . String ( "deploy" , "" , ` Destination to deploy the archive (usually "trunk") ` )
upload = flag . String ( "upload" , "" , ` Destination to upload the archives (usually "gethstore/builds") ` )
)
flag . CommandLine . Parse ( cmdline )
env := build . Env ( )
// Build the iOS XCode framework
build . MustRun ( goTool ( "get" , "golang.org/x/mobile/cmd/gomobile" ) )
2016-11-25 12:34:19 +02:00
build . MustRun ( gomobileTool ( "init" ) )
2017-01-05 14:32:30 +02:00
bind := gomobileTool ( "bind" , "--target" , "ios" , "--tags" , "ios" , "-v" , "github.com/ethereum/go-ethereum/mobile" )
2016-11-09 11:13:37 +02:00
2016-11-25 12:34:19 +02:00
if * local {
// If we're building locally, use the build folder and stop afterwards
bind . Dir , _ = filepath . Abs ( GOBIN )
build . MustRun ( bind )
return
}
2016-11-09 11:13:37 +02:00
archive := "geth-" + archiveBasename ( "ios" , env )
if err := os . Mkdir ( archive , os . ModePerm ) ; err != nil {
log . Fatal ( err )
2016-10-17 15:17:14 +03:00
}
2016-11-09 11:13:37 +02:00
bind . Dir , _ = filepath . Abs ( archive )
build . MustRun ( bind )
build . MustRunCommand ( "tar" , "-zcvf" , archive + ".tar.gz" , archive )
// Skip CocoaPods deploy and Azure upload for PR builds
maybeSkipArchive ( env )
// Sign and upload the framework to Azure
if err := archiveUpload ( archive + ".tar.gz" , * upload , * signer ) ; err != nil {
log . Fatal ( err )
}
// Prepare and upload a PodSpec to CocoaPods
if * deploy != "" {
2016-11-22 18:35:49 +02:00
meta := newPodMetadata ( env , archive )
2016-11-28 12:56:13 +02:00
build . Render ( "build/pod.podspec" , "Geth.podspec" , 0755 , meta )
build . MustRunCommand ( "pod" , * deploy , "push" , "Geth.podspec" , "--allow-warnings" , "--verbose" )
2016-11-09 11:13:37 +02:00
}
}
type podMetadata struct {
Version string
Commit string
2016-11-22 18:35:49 +02:00
Archive string
2016-11-09 11:13:37 +02:00
Contributors [ ] podContributor
2016-10-17 15:17:14 +03:00
}
2016-11-09 11:13:37 +02:00
type podContributor struct {
Name string
Email string
}
2016-11-22 18:35:49 +02:00
func newPodMetadata ( env build . Environment , archive string ) podMetadata {
2016-11-09 11:13:37 +02:00
// Collect the list of authors from the repo root
contribs := [ ] podContributor { }
if authors , err := os . Open ( "AUTHORS" ) ; err == nil {
defer authors . Close ( )
scanner := bufio . NewScanner ( authors )
for scanner . Scan ( ) {
// Skip any whitespace from the authors list
line := strings . TrimSpace ( scanner . Text ( ) )
if line == "" || line [ 0 ] == '#' {
continue
}
// Split the author and insert as a contributor
2016-11-22 18:35:49 +02:00
re := regexp . MustCompile ( "([^<]+) <(.+)>" )
2016-11-09 11:13:37 +02:00
parts := re . FindStringSubmatch ( line )
if len ( parts ) == 3 {
contribs = append ( contribs , podContributor { Name : parts [ 1 ] , Email : parts [ 2 ] } )
}
}
}
2016-11-28 12:56:13 +02:00
version := build . VERSION ( )
2016-11-09 11:13:37 +02:00
if isUnstableBuild ( env ) {
2016-11-28 12:56:13 +02:00
version += "-unstable." + env . Buildnum
2016-11-09 11:13:37 +02:00
}
return podMetadata {
2016-11-22 18:35:49 +02:00
Archive : archive ,
2016-11-28 12:56:13 +02:00
Version : version ,
2016-11-09 11:13:37 +02:00
Commit : env . Commit ,
Contributors : contribs ,
}
2016-11-08 23:55:39 +02:00
}
2016-08-08 13:41:55 +03:00
// Cross compilation
func doXgo ( cmdline [ ] string ) {
2017-03-01 10:47:55 +02:00
var (
alltools = flag . Bool ( "alltools" , false , ` Flag whether we're building all known tools, or only on in particular ` )
)
2016-09-30 22:03:08 +02:00
flag . CommandLine . Parse ( cmdline )
env := build . Env ( )
2016-08-08 13:41:55 +03:00
// Make sure xgo is available for cross compilation
gogetxgo := goTool ( "get" , "github.com/karalabe/xgo" )
build . MustRun ( gogetxgo )
2017-03-01 10:47:55 +02:00
// If all tools building is requested, build everything the builder wants
args := append ( buildFlags ( env ) , flag . Args ( ) ... )
if * alltools {
2017-03-03 12:18:44 +02:00
args = append ( args , [ ] string { "--dest" , GOBIN } ... )
2017-03-01 10:47:55 +02:00
for _ , res := range allToolsArchiveFiles {
if strings . HasPrefix ( res , GOBIN ) {
// Binary tool found, cross build it explicitly
args = append ( args , "./" + filepath . Join ( "cmd" , filepath . Base ( res ) ) )
xgo := xgoTool ( args )
build . MustRun ( xgo )
args = args [ : len ( args ) - 1 ]
}
}
return
}
// Otherwise xxecute the explicit cross compilation
2017-03-03 12:18:44 +02:00
path := args [ len ( args ) - 1 ]
args = append ( args [ : len ( args ) - 1 ] , [ ] string { "--dest" , GOBIN , path } ... )
2017-03-01 10:47:55 +02:00
xgo := xgoTool ( args )
2016-09-30 22:03:08 +02:00
build . MustRun ( xgo )
2016-08-08 13:41:55 +03:00
}
2016-09-30 22:03:08 +02:00
func xgoTool ( args [ ] string ) * exec . Cmd {
2016-08-08 13:41:55 +03:00
cmd := exec . Command ( filepath . Join ( GOBIN , "xgo" ) , args ... )
cmd . Env = [ ] string {
"GOPATH=" + build . GOPATH ( ) ,
"GOBIN=" + GOBIN ,
}
for _ , e := range os . Environ ( ) {
if strings . HasPrefix ( e , "GOPATH=" ) || strings . HasPrefix ( e , "GOBIN=" ) {
continue
}
cmd . Env = append ( cmd . Env , e )
}
return cmd
}