2013-07-08 00:35:10 +00:00
// +build ignore
2013-06-10 22:55:17 +00:00
/ *
Copyright 2013 Google Inc .
Licensed under the Apache License , Version 2.0 ( the "License" ) ;
you may not use this file except in compliance with the License .
You may obtain a copy of the License at
http : //www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing , software
distributed under the License is distributed on an "AS IS" BASIS ,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND , either express or implied .
See the License for the specific language governing permissions and
limitations under the License .
* /
// This program builds Camlistore.
//
2013-06-11 11:19:46 +00:00
// $ go run make.go
//
// See the BUILDING file.
//
2013-06-10 22:55:17 +00:00
// The output binaries go into the ./bin/ directory (under the
// Camlistore root, where make.go is)
package main
import (
2013-06-18 20:02:03 +00:00
"archive/zip"
"bytes"
2013-06-10 22:55:17 +00:00
"flag"
"fmt"
"io"
"io/ioutil"
"log"
"os"
"os/exec"
2013-09-21 11:45:36 +00:00
pathpkg "path"
2013-06-10 22:55:17 +00:00
"path/filepath"
2013-06-20 13:49:34 +00:00
"regexp"
2013-06-10 22:55:17 +00:00
"runtime"
2013-08-26 22:52:02 +00:00
"strconv"
2013-06-10 22:55:17 +00:00
"strings"
2013-06-19 06:14:36 +00:00
"time"
2013-06-10 22:55:17 +00:00
)
2013-08-25 17:57:51 +00:00
var haveSQLite = checkHaveSQLite ( )
2013-06-10 23:18:29 +00:00
var (
2013-12-29 05:27:28 +00:00
embedResources = flag . Bool ( "embed_static" , true , "Whether to embed resources needed by the UI such as images, css, and javascript." )
2013-08-26 22:52:02 +00:00
sqlFlag = flag . String ( "sqlite" , "auto" , "Whether you want SQLite in your build: yes, no, or auto." )
2013-06-18 20:02:03 +00:00
all = flag . Bool ( "all" , false , "Force rebuild of everything (go install -a)" )
2013-12-11 11:53:36 +00:00
race = flag . Bool ( "race" , false , "Build race-detector version of binaries (they will run slowly)" )
2013-06-18 20:02:03 +00:00
verbose = flag . Bool ( "v" , false , "Verbose mode" )
2014-02-20 04:56:05 +00:00
targets = flag . String ( "targets" , "" , "Optional comma-separated list of targets (i.e go packages) to build and install. '*' builds everything. Empty builds defaults for this platform. Example: camlistore.org/server/camlistored,camlistore.org/cmd/camput" )
2013-06-29 20:52:01 +00:00
quiet = flag . Bool ( "quiet" , false , "Don't print anything unless there's a failure." )
2013-09-11 17:51:07 +00:00
onlysync = flag . Bool ( "onlysync" , false , "Only populate the temporary source/build tree and output its full path. It is meant to prepare the environment for running the full test suite with 'devcam test'." )
2013-08-26 13:09:57 +00:00
// TODO(mpl): looks like ifModsSince is not used anywhere?
ifModsSince = flag . Int64 ( "if_mods_since" , 0 , "If non-zero return immediately without building if there aren't any filesystem modifications past this time (in unix seconds)" )
buildARCH = flag . String ( "arch" , runtime . GOARCH , "Architecture to build for." )
buildOS = flag . String ( "os" , runtime . GOOS , "Operating system to build for." )
2013-06-10 23:18:29 +00:00
)
2013-06-10 22:55:17 +00:00
2013-06-20 13:49:34 +00:00
var (
// buildGoPath becomes our child "go" processes' GOPATH environment variable
buildGoPath string
// Our temporary source tree root and build dir, i.e: buildGoPath + "src/camlistore.org"
buildSrcDir string
// files mirrored from camRoot to buildSrcDir
2014-05-15 03:59:13 +00:00
rxMirrored = regexp . MustCompile ( ` ^([a-zA-Z0-9\-\_]+\.(?:blobs|camli|css|eot|err|gif|go|gpg|html|ico|jpg|js|json|xml|min\.css|min\.js|mp3|otf|png|svg|pdf|psd|tiff|ttf|woff|xcf|tar\.gz|gz|tar\.xz|tbz2|zip))$ ` )
2013-06-20 13:49:34 +00:00
)
2013-06-10 22:55:17 +00:00
func main ( ) {
log . SetFlags ( 0 )
flag . Parse ( )
camRoot , err := os . Getwd ( )
if err != nil {
log . Fatalf ( "Failed to get current directory: %v" , err )
}
verifyCamlistoreRoot ( camRoot )
2013-08-26 22:52:02 +00:00
cross := runtime . GOOS != * buildOS || runtime . GOARCH != * buildARCH
var sql bool
if * sqlFlag == "auto" {
sql = ! cross && haveSQLite
} else {
sql , err = strconv . ParseBool ( * sqlFlag )
if err != nil {
log . Fatalf ( "Bad boolean --sql flag %q" , * sqlFlag )
2013-07-05 17:41:07 +00:00
}
}
2013-08-26 22:52:02 +00:00
if cross && sql {
log . Fatalf ( "SQLite isn't available when cross-compiling to another OS. Set --sqlite=false." )
}
if sql && ! haveSQLite {
2013-09-19 09:17:57 +00:00
log . Printf ( "SQLite not found. Either install it, or run make.go with --sqlite=false See https://code.google.com/p/camlistore/wiki/SQLite" )
2013-08-25 17:57:51 +00:00
switch runtime . GOOS {
case "darwin" :
log . Printf ( "On OS X, run 'brew install sqlite3 pkg-config'. Get brew from http://mxcl.github.io/homebrew/" )
case "linux" :
log . Printf ( "On Linux, run 'sudo apt-get install libsqlite3-dev' or equivalent." )
case "windows" :
log . Printf ( "SQLite is not easy on windows. Please see http://camlistore.org/docs/server-config#windows" )
}
os . Exit ( 2 )
}
2013-06-10 23:22:28 +00:00
buildBaseDir := "build-gopath"
2013-08-26 22:52:02 +00:00
if ! sql {
2013-06-10 23:22:28 +00:00
buildBaseDir += "-nosqlite"
}
2013-06-20 13:49:34 +00:00
buildGoPath = filepath . Join ( camRoot , "tmp" , buildBaseDir )
2013-06-10 22:55:17 +00:00
binDir := filepath . Join ( camRoot , "bin" )
2013-06-20 13:49:34 +00:00
buildSrcDir = filepath . Join ( buildGoPath , "src" , "camlistore.org" )
2013-06-10 22:55:17 +00:00
2013-06-20 13:49:34 +00:00
if err := os . MkdirAll ( buildSrcDir , 0755 ) ; err != nil {
2013-06-10 22:55:17 +00:00
log . Fatal ( err )
}
version := getVersion ( camRoot )
2013-06-10 23:22:28 +00:00
if * verbose {
log . Printf ( "Camlistore version = %s" , version )
2013-08-26 22:52:02 +00:00
log . Printf ( "SQLite included: %v" , sql )
2013-06-20 13:49:34 +00:00
log . Printf ( "Temporary source: %s" , buildSrcDir )
2013-06-10 23:22:28 +00:00
log . Printf ( "Output binaries: %s" , binDir )
}
2013-06-10 22:55:17 +00:00
2013-09-11 17:51:07 +00:00
// TODO(mpl): main is getting long. We could probably move all the mirroring
// dance to its own func.
2013-06-20 13:49:34 +00:00
// We copy all *.go files from camRoot's goDirs to buildSrcDir.
2014-05-08 14:07:29 +00:00
goDirs := [ ] string { "app" , "cmd" , "pkg" , "dev" , "server/camlistored" , "third_party" }
2013-09-11 17:51:07 +00:00
if * onlysync {
goDirs = append ( goDirs , "server/appengine" , "config" )
}
2013-06-10 22:55:17 +00:00
// Copy files we do want in our mirrored GOPATH. This has the side effect of
// populating wantDestFile, populated by mirrorFile.
2013-06-29 20:52:01 +00:00
var latestSrcMod time . Time
2013-06-10 22:55:17 +00:00
for _ , dir := range goDirs {
2013-06-20 13:49:34 +00:00
oriPath := filepath . Join ( camRoot , filepath . FromSlash ( dir ) )
dstPath := buildSrcPath ( dir )
2013-12-25 01:42:32 +00:00
if maxMod , err := mirrorDir ( oriPath , dstPath , mirrorOpts { sqlite : sql } ) ; err != nil {
2013-06-20 13:49:34 +00:00
log . Fatalf ( "Error while mirroring %s to %s: %v" , oriPath , dstPath , err )
2013-06-29 20:52:01 +00:00
} else {
if maxMod . After ( latestSrcMod ) {
latestSrcMod = maxMod
}
2013-06-10 22:55:17 +00:00
}
}
2013-06-29 20:52:01 +00:00
verifyGoVersion ( )
2013-09-11 17:51:07 +00:00
if * onlysync {
mirrorFile ( "make.go" , filepath . Join ( buildSrcDir , "make.go" ) )
deleteUnwantedOldMirrorFiles ( buildSrcDir , true )
fmt . Println ( buildGoPath )
return
}
2014-02-21 21:59:26 +00:00
buildAll := false
2013-09-11 05:42:24 +00:00
targs := [ ] string {
"camlistore.org/dev/devcam" ,
"camlistore.org/cmd/camget" ,
"camlistore.org/cmd/camput" ,
"camlistore.org/cmd/camtool" ,
"camlistore.org/server/camlistored" ,
2014-05-08 14:07:29 +00:00
"camlistore.org/app/hello" ,
2014-06-14 20:14:34 +00:00
"camlistore.org/app/publisher" ,
2013-09-11 05:42:24 +00:00
}
2014-02-20 04:56:05 +00:00
switch * targets {
case "*" :
buildAll = true
case "" :
2014-02-21 21:59:26 +00:00
// Add cammount to default build targets on OSes that support FUSE.
switch * buildOS {
case "linux" , "darwin" :
targs = append ( targs , "camlistore.org/cmd/cammount" )
}
2014-02-20 04:56:05 +00:00
default :
2013-09-11 05:42:24 +00:00
if t := strings . Split ( * targets , "," ) ; len ( t ) != 0 {
targs = t
}
}
withCamlistored := stringListContains ( targs , "camlistore.org/server/camlistored" )
if * embedResources && withCamlistored {
2013-06-29 20:52:01 +00:00
if * verbose {
log . Printf ( "Embedding resources..." )
}
closureEmbed := buildSrcPath ( "server/camlistored/ui/closure/z_data.go" )
2013-06-20 13:49:34 +00:00
closureSrcDir := filepath . Join ( camRoot , filepath . FromSlash ( "third_party/closure/lib" ) )
2013-06-18 20:02:03 +00:00
err := embedClosure ( closureSrcDir , closureEmbed )
if err != nil {
log . Fatal ( err )
}
2013-06-18 21:45:40 +00:00
wantDestFile [ closureEmbed ] = true
2013-06-20 13:49:34 +00:00
if err = buildGenfileembed ( ) ; err != nil {
log . Fatal ( err )
}
if err = genEmbeds ( ) ; err != nil {
2013-06-18 23:04:27 +00:00
log . Fatal ( err )
}
2013-06-18 20:02:03 +00:00
}
2013-09-11 05:42:24 +00:00
deleteUnwantedOldMirrorFiles ( buildSrcDir , withCamlistored )
2013-06-18 21:45:40 +00:00
2013-06-10 22:55:17 +00:00
tags := ""
2013-08-26 22:52:02 +00:00
if sql {
2013-06-10 22:55:17 +00:00
tags = "with_sqlite"
}
2013-06-22 03:18:02 +00:00
baseArgs := [ ] string { "install" , "-v" }
2013-06-10 23:18:29 +00:00
if * all {
2013-06-22 03:18:02 +00:00
baseArgs = append ( baseArgs , "-a" )
2013-06-10 23:18:29 +00:00
}
2013-12-11 11:53:36 +00:00
if * race {
baseArgs = append ( baseArgs , "-race" )
}
2013-06-22 03:18:02 +00:00
baseArgs = append ( baseArgs ,
2013-06-10 22:55:17 +00:00
"--ldflags=-X camlistore.org/pkg/buildinfo.GitInfo " + version ,
2013-06-22 03:18:02 +00:00
"--tags=" + tags )
2013-09-21 11:45:36 +00:00
// First install command: build just the final binaries, installed to a GOBIN
// under <camlistore_root>/bin:
args := append ( baseArgs , targs ... )
if buildAll {
args = append ( args ,
2014-05-08 14:07:29 +00:00
"camlistore.org/app/..." ,
2013-09-21 11:45:36 +00:00
"camlistore.org/pkg/..." ,
"camlistore.org/server/..." ,
"camlistore.org/third_party/..." ,
)
}
2013-06-10 23:18:29 +00:00
cmd := exec . Command ( "go" , args ... )
2013-06-18 22:21:56 +00:00
cmd . Env = append ( cleanGoEnv ( ) ,
2013-06-20 13:49:34 +00:00
"GOPATH=" + buildGoPath ,
2013-06-18 22:21:56 +00:00
)
2013-06-29 20:52:01 +00:00
var output bytes . Buffer
if * quiet {
cmd . Stdout = & output
cmd . Stderr = & output
} else {
cmd . Stdout = os . Stdout
cmd . Stderr = os . Stderr
}
2013-06-10 23:22:28 +00:00
if * verbose {
2013-07-05 17:41:07 +00:00
log . Printf ( "Running go install of main binaries with args %s" , cmd . Args )
2013-06-22 03:18:02 +00:00
}
if err := cmd . Run ( ) ; err != nil {
2013-07-04 02:44:57 +00:00
log . Fatalf ( "Error building main binaries: %v\n%s" , err , output . String ( ) )
2013-06-10 23:22:28 +00:00
}
2013-06-29 20:52:01 +00:00
2013-09-21 11:45:36 +00:00
// Copy the binaries from $CAMROOT/tmp/build-gopath-foo/bin to $CAMROOT/bin.
// This is necessary (instead of just using GOBIN environment variable) so
// each tmp/build-gopath-* has its own binary modtimes for its own build tags.
// Otherwise switching sqlite true<->false doesn't necessarily cause a rebuild.
// See camlistore.org/issue/229
for _ , targ := range targs {
src := exeName ( filepath . Join ( actualBinDir ( filepath . Join ( buildGoPath , "bin" ) ) , pathpkg . Base ( targ ) ) )
dst := exeName ( filepath . Join ( actualBinDir ( binDir ) , pathpkg . Base ( targ ) ) )
if err := mirrorFile ( src , dst ) ; err != nil {
log . Fatalf ( "Error copying %s to %s: %v" , src , dst , err )
2013-06-29 20:52:01 +00:00
}
2013-06-10 22:55:17 +00:00
}
2013-06-22 03:18:02 +00:00
2013-06-29 20:52:01 +00:00
if ! * quiet {
2013-09-04 20:44:43 +00:00
log . Printf ( "Success. Binaries are in %s" , actualBinDir ( binDir ) )
2013-06-29 20:52:01 +00:00
}
2013-06-10 22:55:17 +00:00
}
2013-09-04 02:48:32 +00:00
func actualBinDir ( dir string ) string {
if * buildARCH == runtime . GOARCH && * buildOS == runtime . GOOS {
return dir
}
return filepath . Join ( dir , * buildOS + "_" + * buildARCH )
}
2013-07-11 09:49:54 +00:00
// Create an environment variable of the form key=value.
func envPair ( key , value string ) string {
return fmt . Sprintf ( "%s=%s" , key , value )
}
2013-06-18 22:21:56 +00:00
// cleanGoEnv returns a copy of the current environment with GOPATH and GOBIN removed.
2013-08-20 14:46:32 +00:00
// it also sets GOOS and GOARCH as needed when cross-compiling.
2013-06-18 22:21:56 +00:00
func cleanGoEnv ( ) ( clean [ ] string ) {
for _ , env := range os . Environ ( ) {
if strings . HasPrefix ( env , "GOPATH=" ) || strings . HasPrefix ( env , "GOBIN=" ) {
continue
}
2013-08-20 14:46:32 +00:00
// We skip these two as well, otherwise they'd take precedence over the
// ones appended below.
if * buildOS != runtime . GOOS && strings . HasPrefix ( env , "GOOS=" ) {
continue
}
if * buildARCH != runtime . GOARCH && strings . HasPrefix ( env , "GOARCH=" ) {
continue
}
2013-06-18 22:21:56 +00:00
clean = append ( clean , env )
}
2013-07-05 17:41:07 +00:00
if * buildOS != runtime . GOOS {
2013-07-11 09:49:54 +00:00
clean = append ( clean , envPair ( "GOOS" , * buildOS ) )
}
if * buildARCH != runtime . GOARCH {
clean = append ( clean , envPair ( "GOARCH" , * buildARCH ) )
2013-07-05 17:41:07 +00:00
}
2013-06-18 22:21:56 +00:00
return
}
2013-07-11 09:49:54 +00:00
// setEnv sets the given key & value in the provided environment.
// Each value in the env list should be of the form key=value.
func setEnv ( env [ ] string , key , value string ) [ ] string {
for i , s := range env {
if strings . HasPrefix ( s , fmt . Sprintf ( "%s=" , key ) ) {
env [ i ] = envPair ( key , value )
return env
}
}
env = append ( env , envPair ( key , value ) )
return env
}
2013-09-11 05:42:24 +00:00
func stringListContains ( strs [ ] string , str string ) bool {
for _ , s := range strs {
if s == str {
return true
}
}
return false
}
2013-06-20 13:49:34 +00:00
// buildSrcPath returns the full path concatenation
// of buildSrcDir with fromSrc.
func buildSrcPath ( fromSrc string ) string {
return filepath . Join ( buildSrcDir , filepath . FromSlash ( fromSrc ) )
}
// genEmbeds generates from the static resources the zembed.*.go
// files that will allow for these resources to be included in
// the camlistored binary.
// It also populates wantDestFile with those files so they're
// kept in between runs.
func genEmbeds ( ) error {
2014-03-28 17:19:09 +00:00
cmdName := exeName ( filepath . Join ( buildGoPath , "bin" , "genfileembed" ) )
2014-06-14 20:14:34 +00:00
for _ , embeds := range [ ] string { "server/camlistored/ui" , "pkg/server" , "third_party/react" , "third_party/glitch" , "third_party/fontawesome" , "app/publisher" } {
2014-04-05 05:59:43 +00:00
embeds := buildSrcPath ( embeds )
2013-06-20 13:49:34 +00:00
args := [ ] string { embeds }
cmd := exec . Command ( cmdName , args ... )
cmd . Env = append ( cleanGoEnv ( ) ,
"GOPATH=" + buildGoPath ,
)
cmd . Stdout = os . Stdout
cmd . Stderr = os . Stderr
if * verbose {
log . Printf ( "Running %s %s" , cmdName , embeds )
}
if err := cmd . Run ( ) ; err != nil {
return fmt . Errorf ( "Error running %s %s: %v" , cmdName , embeds , err )
}
// We mark all the zembeds in builddir as wanted, so that we do not
// have to regen them next time, unless they need updating.
2014-04-06 02:34:13 +00:00
if err := filepath . Walk ( embeds , func ( path string , _ os . FileInfo , err error ) error {
if strings . HasPrefix ( filepath . Base ( path ) , "zembed_" ) {
wantDestFile [ path ] = true
}
2013-06-20 13:49:34 +00:00
return err
2014-04-06 02:34:13 +00:00
} ) ; err != nil {
2013-06-20 13:49:34 +00:00
return err
}
}
return nil
}
func buildGenfileembed ( ) error {
2013-06-18 23:04:27 +00:00
args := [ ] string { "install" , "-v" }
if * all {
args = append ( args , "-a" )
}
args = append ( args ,
2013-06-20 13:49:34 +00:00
filepath . FromSlash ( "camlistore.org/pkg/fileembed/genfileembed" ) ,
2013-06-18 23:04:27 +00:00
)
cmd := exec . Command ( "go" , args ... )
2013-07-11 09:49:54 +00:00
2013-06-18 23:04:27 +00:00
// We don't even need to set GOBIN as it defaults to $GOPATH/bin
// and that is where we want genfileembed to go.
2013-07-11 09:49:54 +00:00
// Here we replace the GOOS and GOARCH valuesfrom the env with the host OS,
// to support cross-compiling.
cmd . Env = cleanGoEnv ( )
cmd . Env = setEnv ( cmd . Env , "GOPATH" , buildGoPath )
cmd . Env = setEnv ( cmd . Env , "GOOS" , runtime . GOOS )
cmd . Env = setEnv ( cmd . Env , "GOARCH" , runtime . GOARCH )
2013-06-18 23:04:27 +00:00
cmd . Stdout = os . Stdout
cmd . Stderr = os . Stderr
if * verbose {
log . Printf ( "Running go with args %s" , args )
}
if err := cmd . Run ( ) ; err != nil {
return fmt . Errorf ( "Error building genfileembed: %v" , err )
}
if * verbose {
2013-06-20 13:49:34 +00:00
log . Printf ( "genfileembed installed in %s" , filepath . Join ( buildGoPath , "bin" ) )
2013-06-18 23:04:27 +00:00
}
return nil
}
2013-06-10 22:55:17 +00:00
// getVersion returns the version of Camlistore. Either from a VERSION file at the root,
// or from git.
func getVersion ( camRoot string ) string {
slurp , err := ioutil . ReadFile ( filepath . Join ( camRoot , "VERSION" ) )
if err == nil {
return strings . TrimSpace ( string ( slurp ) )
}
2013-07-29 01:15:04 +00:00
return gitVersion ( camRoot )
}
var gitVersionRx = regexp . MustCompile ( ` \b\d\d\d\d-\d\d-\d\d-[0-9a-f] { 7,7}\b ` )
// gitVersion returns the git version of the git repo at camRoot as a
// string of the form "yyyy-mm-dd-xxxxxxx", with an optional trailing
// '+' if there are any local uncomitted modifications to the tree.
func gitVersion ( camRoot string ) string {
cmd := exec . Command ( "git" , "rev-list" , "--max-count=1" , "--pretty=format:'%ad-%h'" , "--date=short" , "HEAD" )
cmd . Dir = camRoot
out , err := cmd . Output ( )
2013-06-10 22:55:17 +00:00
if err != nil {
2013-07-29 01:15:04 +00:00
log . Fatalf ( "Error running git rev-list in %s: %v" , camRoot , err )
}
v := strings . TrimSpace ( string ( out ) )
if m := gitVersionRx . FindStringSubmatch ( v ) ; m != nil {
v = m [ 0 ]
} else {
panic ( "Failed to find git version in " + v )
}
cmd = exec . Command ( "git" , "diff" , "--exit-code" )
cmd . Dir = camRoot
if err := cmd . Run ( ) ; err != nil {
v += "+"
2013-06-10 22:55:17 +00:00
}
2013-07-29 01:15:04 +00:00
return v
2013-06-10 22:55:17 +00:00
}
// verifyCamlistoreRoot crashes if dir isn't the Camlistore root directory.
func verifyCamlistoreRoot ( dir string ) {
2013-08-04 02:54:30 +00:00
testFile := filepath . Join ( dir , "pkg" , "blob" , "ref.go" )
2013-06-10 22:55:17 +00:00
if _ , err := os . Stat ( testFile ) ; err != nil {
log . Fatalf ( "make.go must be run from the Camlistore src root directory (where make.go is). Current working directory is %s" , dir )
}
}
func verifyGoVersion ( ) {
_ , err := exec . LookPath ( "go" )
if err != nil {
log . Fatalf ( "Go doesn't appeared to be installed ('go' isn't in your PATH). Install Go 1.1 or newer." )
}
out , err := exec . Command ( "go" , "version" ) . Output ( )
if err != nil {
log . Fatalf ( "Error checking Go version with the 'go' command: %v" , err )
}
fields := strings . Fields ( string ( out ) )
if len ( fields ) < 3 || ! strings . HasPrefix ( string ( out ) , "go version " ) {
log . Fatalf ( "Unexpected output while checking 'go version': %q" , out )
}
version := fields [ 2 ]
switch version {
case "go1" , "go1.0.1" , "go1.0.2" , "go1.0.3" :
2013-10-08 18:47:45 +00:00
log . Fatalf ( "Your version of Go (%s) is too old. Camlistore requires Go 1.1 or later." , version )
2013-06-10 22:55:17 +00:00
}
}
2013-12-25 01:42:32 +00:00
type mirrorOpts struct {
sqlite bool // want sqlite package?
}
func mirrorDir ( src , dst string , opts mirrorOpts ) ( maxMod time . Time , err error ) {
2013-06-29 20:52:01 +00:00
err = filepath . Walk ( src , func ( path string , fi os . FileInfo , err error ) error {
2013-06-10 22:55:17 +00:00
if err != nil {
return err
}
base := fi . Name ( )
if fi . IsDir ( ) {
2013-12-25 01:42:32 +00:00
if ! opts . sqlite && strings . Contains ( path , "mattn" ) && strings . Contains ( path , "go-sqlite3" ) {
return filepath . SkipDir
}
2013-09-11 17:51:07 +00:00
return nil
2013-06-10 22:55:17 +00:00
}
2013-09-11 17:51:07 +00:00
if strings . HasPrefix ( base , ".#" ) || ! rxMirrored . MatchString ( base ) {
2013-06-10 22:55:17 +00:00
return nil
}
suffix , err := filepath . Rel ( src , path )
if err != nil {
return fmt . Errorf ( "Failed to find Rel(%q, %q): %v" , src , path , err )
}
2013-06-29 20:52:01 +00:00
if t := fi . ModTime ( ) ; t . After ( maxMod ) {
maxMod = t
}
2013-06-10 22:55:17 +00:00
return mirrorFile ( path , filepath . Join ( dst , suffix ) )
} )
2013-06-29 20:52:01 +00:00
return
2013-06-10 22:55:17 +00:00
}
var wantDestFile = make ( map [ string ] bool ) // full dest filename => true
2013-09-21 12:23:07 +00:00
func isExecMode ( mode os . FileMode ) bool {
return ( mode & 0111 ) != 0
}
2013-06-10 22:55:17 +00:00
func mirrorFile ( src , dst string ) error {
wantDestFile [ dst ] = true
sfi , err := os . Stat ( src )
if err != nil {
return err
}
if sfi . Mode ( ) & os . ModeType != 0 {
log . Fatalf ( "mirrorFile can't deal with non-regular file %s" , src )
}
dfi , err := os . Stat ( dst )
if err == nil &&
2013-09-21 12:23:07 +00:00
isExecMode ( sfi . Mode ( ) ) == isExecMode ( dfi . Mode ( ) ) &&
2013-06-10 22:55:17 +00:00
( dfi . Mode ( ) & os . ModeType == 0 ) &&
dfi . Size ( ) == sfi . Size ( ) &&
dfi . ModTime ( ) . Unix ( ) == sfi . ModTime ( ) . Unix ( ) {
// Seems to not be modified.
return nil
}
dstDir := filepath . Dir ( dst )
if err := os . MkdirAll ( dstDir , 0755 ) ; err != nil {
return err
}
df , err := os . Create ( dst )
if err != nil {
return err
}
sf , err := os . Open ( src )
if err != nil {
return err
}
defer sf . Close ( )
n , err := io . Copy ( df , sf )
if err == nil && n != sfi . Size ( ) {
err = fmt . Errorf ( "copied wrong size for %s -> %s: copied %d; want %d" , src , dst , n , sfi . Size ( ) )
}
cerr := df . Close ( )
if err == nil {
err = cerr
}
2013-09-21 12:23:07 +00:00
if err == nil {
err = os . Chmod ( dst , sfi . Mode ( ) )
}
2013-06-10 23:18:29 +00:00
if err == nil {
err = os . Chtimes ( dst , sfi . ModTime ( ) , sfi . ModTime ( ) )
}
2013-06-10 22:55:17 +00:00
return err
}
2013-09-11 05:42:24 +00:00
func deleteUnwantedOldMirrorFiles ( dir string , withCamlistored bool ) {
2013-06-10 22:55:17 +00:00
filepath . Walk ( dir , func ( path string , fi os . FileInfo , err error ) error {
if err != nil {
log . Fatalf ( "Error stating while cleaning %s: %v" , path , err )
}
if fi . IsDir ( ) {
return nil
}
if ! wantDestFile [ path ] {
2013-09-11 05:42:24 +00:00
if ! withCamlistored && ( strings . Contains ( path , "zembed_" ) || strings . Contains ( path , "z_data.go" ) ) {
// If we're not building the camlistored binary,
// no need to clean up the embedded Closure, JS,
// CSS, HTML, etc. Doing so would just mean we'd
// have to put it back into place later.
return nil
}
2013-06-29 20:52:01 +00:00
if ! * quiet {
log . Printf ( "Deleting old file from temp build dir: %s" , path )
}
2013-06-10 22:55:17 +00:00
return os . Remove ( path )
}
return nil
} )
}
2013-08-25 17:57:51 +00:00
func checkHaveSQLite ( ) bool {
2013-06-10 22:55:17 +00:00
if runtime . GOOS == "windows" {
// TODO: Find some other non-pkg-config way to test, like
// just compiling a small Go program that sees whether
// it's available.
//
// For now:
return false
}
_ , err := exec . LookPath ( "pkg-config" )
if err != nil {
2013-08-25 17:57:51 +00:00
return false
2013-06-10 22:55:17 +00:00
}
2013-12-25 02:24:24 +00:00
out , err := exec . Command ( "pkg-config" , "--libs" , "sqlite3" ) . Output ( )
2013-06-14 20:21:04 +00:00
if err != nil && err . Error ( ) == "exit status 1" {
// This is sloppy (comparing against a string), but
// doing it correctly requires using multiple *.go
// files to portably get the OS-syscall bits, and I
// want to keep make.go a single file.
return false
}
2013-06-10 22:55:17 +00:00
if err != nil {
log . Fatalf ( "Can't determine whether sqlite3 is available, and where. pkg-config error was: %v, %s" , err , out )
}
return strings . TrimSpace ( string ( out ) ) != ""
}
2013-06-18 20:02:03 +00:00
func embedClosure ( closureDir , embedFile string ) error {
if _ , err := os . Stat ( closureDir ) ; err != nil {
return fmt . Errorf ( "Could not stat %v: %v" , closureDir , err )
}
// first, zip it
var zipbuf bytes . Buffer
2013-06-19 01:39:43 +00:00
var zipdest io . Writer = & zipbuf
if os . Getenv ( "CAMLI_WRITE_TMP_ZIP" ) != "" {
f , _ := os . Create ( "/tmp/camli-closure.zip" )
zipdest = io . MultiWriter ( zipdest , f )
defer f . Close ( )
}
2013-06-19 06:14:36 +00:00
var modTime time . Time
2013-06-19 01:39:43 +00:00
w := zip . NewWriter ( zipdest )
2013-06-18 20:02:03 +00:00
err := filepath . Walk ( closureDir , func ( path string , fi os . FileInfo , err error ) error {
if err != nil {
return err
}
suffix , err := filepath . Rel ( closureDir , path )
if err != nil {
return fmt . Errorf ( "Failed to find Rel(%q, %q): %v" , closureDir , path , err )
}
if fi . IsDir ( ) {
return nil
}
2013-06-19 06:14:36 +00:00
if mt := fi . ModTime ( ) ; mt . After ( modTime ) {
modTime = mt
}
2013-06-18 20:02:03 +00:00
b , err := ioutil . ReadFile ( path )
if err != nil {
return err
}
2013-08-04 05:48:09 +00:00
f , err := w . Create ( filepath . ToSlash ( suffix ) )
2013-06-18 20:02:03 +00:00
if err != nil {
log . Fatal ( err )
}
_ , err = f . Write ( b )
return err
} )
if err != nil {
return err
}
err = w . Close ( )
if err != nil {
return err
}
// then embed it as a quoted string
var qb bytes . Buffer
fmt . Fprint ( & qb , "package closure\n\n" )
2013-06-19 06:14:36 +00:00
fmt . Fprint ( & qb , "import \"time\"\n\n" )
fmt . Fprint ( & qb , "func init() {\n" )
fmt . Fprintf ( & qb , "\tZipModTime = time.Unix(%d, 0)\n" , modTime . Unix ( ) )
fmt . Fprint ( & qb , "\tZipData = " )
2013-06-18 20:02:03 +00:00
quote ( & qb , zipbuf . Bytes ( ) )
2013-06-19 06:14:36 +00:00
fmt . Fprint ( & qb , "\n}\n" )
2013-06-18 20:02:03 +00:00
// and write to a .go file
// TODO(mpl): do not regenerate the whole zip file if the modtime
// of the z_data.go file is greater than the modtime of all the closure *.js files.
if err := writeFileIfDifferent ( embedFile , qb . Bytes ( ) ) ; err != nil {
return err
}
return nil
}
func writeFileIfDifferent ( filename string , contents [ ] byte ) error {
fi , err := os . Stat ( filename )
if err == nil && fi . Size ( ) == int64 ( len ( contents ) ) && contentsEqual ( filename , contents ) {
return nil
}
return ioutil . WriteFile ( filename , contents , 0644 )
}
func contentsEqual ( filename string , contents [ ] byte ) bool {
got , err := ioutil . ReadFile ( filename )
if err != nil {
return false
}
return bytes . Equal ( got , contents )
}
// quote escapes and quotes the bytes from bs and writes
// them to dest.
func quote ( dest * bytes . Buffer , bs [ ] byte ) {
dest . WriteByte ( '"' )
for _ , b := range bs {
if b == '\n' {
dest . WriteString ( ` \n ` )
2013-06-19 01:39:43 +00:00
continue
2013-06-18 20:02:03 +00:00
}
if b == '\\' {
dest . WriteString ( ` \\ ` )
continue
}
if b == '"' {
dest . WriteString ( ` \" ` )
continue
}
if ( b >= 32 && b <= 126 ) || b == '\t' {
dest . WriteByte ( b )
continue
}
fmt . Fprintf ( dest , "\\x%02x" , b )
}
dest . WriteByte ( '"' )
}
2013-09-21 11:45:36 +00:00
func exeName ( s string ) string {
if * buildOS == "windows" {
return s + ".exe"
}
return s
}