291 lines
7.6 KiB
Go
291 lines
7.6 KiB
Go
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"
|
|
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: "
|
|
)
|