package lfs import ( "fmt" "io/ioutil" "os" "path/filepath" "runtime" "strings" "github.com/github/git-lfs/vendor/_nuts/github.com/rubyist/tracerx" ) const ( Version = "0.6.0-pre" tempDirPerms = 0755 localMediaDirPerms = 0755 localLogDirPerms = 0755 ) var ( LargeSizeThreshold = 5 * 1024 * 1024 TempDir = filepath.Join(os.TempDir(), "git-lfs") GitCommit string UserAgent string LocalWorkingDir string LocalGitDir string // parent of index / config / hooks etc LocalGitStorageDir string // parent of objects/lfs (may be same as LocalGitDir but may not) LocalMediaDir string // root of lfs objects LocalLogDir string checkedTempDir string ) func TempFile(prefix string) (*os.File, error) { if checkedTempDir != TempDir { if err := os.MkdirAll(TempDir, tempDirPerms); err != nil { return nil, err } checkedTempDir = TempDir } return ioutil.TempFile(TempDir, prefix) } func ResetTempDir() error { checkedTempDir = "" return os.RemoveAll(TempDir) } func localMediaDirNoCreate(sha string) string { return filepath.Join(LocalMediaDir, sha[0:2], sha[2:4]) } func localMediaPathNoCreate(sha string) string { return filepath.Join(localMediaDirNoCreate(sha), sha) } func LocalMediaPath(sha string) (string, error) { path := localMediaDirNoCreate(sha) if err := os.MkdirAll(path, localMediaDirPerms); err != nil { return "", fmt.Errorf("Error trying to create local media directory in '%s': %s", path, err) } return filepath.Join(path, sha), nil } func ObjectExistsOfSize(sha string, size int64) bool { path := localMediaPathNoCreate(sha) return FileExistsOfSize(path, size) } func Environ() []string { osEnviron := os.Environ() env := make([]string, 0, len(osEnviron)+7) env = append(env, fmt.Sprintf("LocalWorkingDir=%s", LocalWorkingDir), fmt.Sprintf("LocalGitDir=%s", LocalGitDir), fmt.Sprintf("LocalGitStorageDir=%s", LocalGitStorageDir), fmt.Sprintf("LocalMediaDir=%s", LocalMediaDir), fmt.Sprintf("TempDir=%s", TempDir), fmt.Sprintf("ConcurrentTransfers=%d", Config.ConcurrentTransfers()), fmt.Sprintf("BatchTransfer=%v", Config.BatchTransfer()), ) for _, e := range osEnviron { if !strings.Contains(e, "GIT_") { continue } env = append(env, e) } return env } func InRepo() bool { return LocalWorkingDir != "" } func ResolveDirs() { var err error LocalWorkingDir, LocalGitDir, err = resolveGitDir() if err == nil { LocalGitStorageDir = resolveGitStorageDir(LocalGitDir) LocalMediaDir = filepath.Join(LocalGitStorageDir, "lfs", "objects") LocalLogDir = filepath.Join(LocalMediaDir, "logs") TempDir = filepath.Join(LocalGitDir, "lfs", "tmp") // temp files per worktree if err := os.MkdirAll(LocalMediaDir, localMediaDirPerms); err != nil { panic(fmt.Errorf("Error trying to create objects directory in '%s': %s", LocalMediaDir, err)) } if err := os.MkdirAll(LocalLogDir, localLogDirPerms); err != nil { panic(fmt.Errorf("Error trying to create log directory in '%s': %s", LocalLogDir, err)) } if err := os.MkdirAll(TempDir, tempDirPerms); err != nil { panic(fmt.Errorf("Error trying to create temp directory in '%s': %s", TempDir, err)) } } } func init() { tracerx.DefaultKey = "GIT" tracerx.Prefix = "trace git-lfs: " ResolveDirs() gitCommit := "" if len(GitCommit) > 0 { gitCommit = "; git " + GitCommit } UserAgent = fmt.Sprintf("git-lfs/%s (GitHub; %s %s; go %s%s)", Version, runtime.GOOS, runtime.GOARCH, strings.Replace(runtime.Version(), "go", "", 1), gitCommit, ) } func resolveGitDir() (string, string, error) { gitDir := Config.Getenv("GIT_DIR") workTree := Config.Getenv("GIT_WORK_TREE") if gitDir != "" { return processGitDirVar(gitDir, workTree) } wd, err := os.Getwd() if err != nil { return "", "", err } workTreeR, gitDirR, err := recursiveResolveGitDir(wd) if err != nil { return "", "", err } if workTree != "" { return processWorkTreeVar(gitDirR, workTree) } return workTreeR, gitDirR, nil } func processGitDirVar(gitDir, workTree string) (string, string, error) { if workTree != "" { return processWorkTreeVar(gitDir, workTree) } // See `core.worktree` in `man git-config`: // “If --git-dir or GIT_DIR is specified but none of --work-tree, GIT_WORK_TREE and // core.worktree is specified, the current working directory is regarded as the top // level of your working tree.” wd, err := os.Getwd() if err != nil { return "", "", err } return wd, gitDir, nil } func processWorkTreeVar(gitDir, workTree string) (string, string, error) { // See `core.worktree` in `man git-config`: // “The value [of core.worktree, GIT_WORK_TREE, or --work-tree] can be an absolute path // or relative to the path to the .git directory, which is either specified // by --git-dir or GIT_DIR, or automatically discovered.” if filepath.IsAbs(workTree) { return workTree, gitDir, nil } base := filepath.Dir(filepath.Clean(gitDir)) absWorkTree := filepath.Join(base, workTree) return absWorkTree, gitDir, nil } func recursiveResolveGitDir(dir string) (string, string, error) { var cleanDir = filepath.Clean(dir) if cleanDir[len(cleanDir)-1] == os.PathSeparator { return "", "", fmt.Errorf("Git repository not found") } if filepath.Base(dir) == gitExt { // We're in the `.git` directory. Make no assumptions about the working directory. return "", dir, nil } gitDir := filepath.Join(dir, gitExt) info, err := os.Stat(gitDir) if err != nil { // Found neither a directory nor a file named `.git`. // Move one directory up. return recursiveResolveGitDir(filepath.Dir(dir)) } if !info.IsDir() { // Found a file named `.git` (we're in a submodule). return resolveDotGitFile(gitDir) } // Found the `.git` directory. return dir, gitDir, nil } func resolveDotGitFile(file string) (string, string, error) { // The local working directory is the directory the `.git` file is located in. wd := filepath.Dir(file) // The `.git` file tells us where the submodules `.git` directory is. gitDir, err := processDotGitFile(file) if err != nil { return "", "", err } return wd, gitDir, nil } func processDotGitFile(file string) (string, error) { return processGitRedirectFile(file, gitPtrPrefix) } func processGitRedirectFile(file, prefix string) (string, error) { data, err := ioutil.ReadFile(file) if err != nil { return "", err } contents := string(data) var dir string if len(prefix) > 0 { if !strings.HasPrefix(contents, prefix) { // Prefix required & not found return "", nil } dir = strings.TrimSpace(contents[len(prefix):]) } else { dir = strings.TrimSpace(contents) } if !filepath.IsAbs(dir) { // The .git file contains a relative path. // Create an absolute path based on the directory the .git file is located in. dir = filepath.Join(filepath.Dir(file), dir) } return dir, nil } // From a git dir, get the location that objects are to be stored (we will store lfs alongside) // Sometimes there is an additional level of redirect on the .git folder by way of a commondir file // before you find object storage, e.g. 'git worktree' uses this. It redirects to gitdir either by GIT_DIR // (during setup) or .git/git-dir: (during use), but this only contains the index etc, the objects // are found in another git dir via 'commondir'. func resolveGitStorageDir(gitDir string) string { commondirpath := filepath.Join(gitDir, "commondir") if FileExists(commondirpath) && !DirExists(filepath.Join(gitDir, "objects")) { // no git-dir: prefix in commondir storage, err := processGitRedirectFile(commondirpath, "") if err == nil { return storage } } return gitDir } const ( gitExt = ".git" gitPtrPrefix = "gitdir: " )