package lfs import ( "fmt" "net/http" "net/url" "os" "path/filepath" "strconv" "strings" "sync" "github.com/github/git-lfs/git" "github.com/github/git-lfs/vendor/_nuts/github.com/rubyist/tracerx" ) var ( Config = NewConfig() defaultRemote = "origin" ) type Configuration struct { CurrentRemote string httpClient *HttpClient redirectingHttpClient *http.Client envVars map[string]string isTracingHttp bool isLoggingStats bool loading sync.Mutex // guards initialization of gitConfig and remotes gitConfig map[string]string remotes []string } func NewConfig() *Configuration { c := &Configuration{ CurrentRemote: defaultRemote, envVars: make(map[string]string), } c.isTracingHttp = c.GetenvBool("GIT_CURL_VERBOSE", false) c.isLoggingStats = c.GetenvBool("GIT_LOG_STATS", false) return c } func (c *Configuration) Getenv(key string) string { if i, ok := c.envVars[key]; ok { return i } v := os.Getenv(key) c.envVars[key] = v return v } func (c *Configuration) Setenv(key, value string) error { // Check see if we have this in our cache, if so update it if _, ok := c.envVars[key]; ok { c.envVars[key] = value } // Now set in process return os.Setenv(key, value) } // GetenvBool parses a boolean environment variable and returns the result as a bool. // If the environment variable is unset, empty, or if the parsing fails, // the value of def (default) is returned instead. func (c *Configuration) GetenvBool(key string, def bool) bool { s := c.Getenv(key) if len(s) == 0 { return def } b, err := strconv.ParseBool(s) if err != nil { return def } return b } func (c *Configuration) Endpoint() Endpoint { if url, ok := c.GitConfig("lfs.url"); ok { return NewEndpoint(url) } if len(c.CurrentRemote) > 0 && c.CurrentRemote != defaultRemote { if endpoint := c.RemoteEndpoint(c.CurrentRemote); len(endpoint.Url) > 0 { return endpoint } } return c.RemoteEndpoint(defaultRemote) } func (c *Configuration) ConcurrentTransfers() int { uploads := 3 if v, ok := c.GitConfig("lfs.concurrenttransfers"); ok { n, err := strconv.Atoi(v) if err == nil && n > 0 { uploads = n } } return uploads } func (c *Configuration) BatchTransfer() bool { if v, ok := c.GitConfig("lfs.batch"); ok { if v == "true" || v == "" { return true } // Any numeric value except 0 is considered true if n, err := strconv.Atoi(v); err == nil && n != 0 { return true } } return false } // PrivateAccess will retrieve the access value and return true if // the value is set to private. When a repo is marked as having private // access, the http requests for the batch api will fetch the credentials // before running, otherwise the request will run without credentials. func (c *Configuration) PrivateAccess() bool { key := fmt.Sprintf("lfs.%s.access", c.Endpoint().Url) if v, ok := c.GitConfig(key); ok { if strings.ToLower(v) == "private" { return true } } return false } // SetPrivateAccess will set the private access flag in .git/config. func (c *Configuration) SetPrivateAccess() { tracerx.Printf("setting repository access to private") key := fmt.Sprintf("lfs.%s.access", c.Endpoint().Url) configFile := filepath.Join(LocalGitDir, "config") git.Config.SetLocal(configFile, key, "private") // Modify the config cache because it's checked again in this process // without being reloaded. c.loading.Lock() c.gitConfig[key] = "private" c.loading.Unlock() } func (c *Configuration) RemoteEndpoint(remote string) Endpoint { if len(remote) == 0 { remote = defaultRemote } if url, ok := c.GitConfig("remote." + remote + ".lfsurl"); ok { return NewEndpoint(url) } if url, ok := c.GitConfig("remote." + remote + ".url"); ok { return NewEndpointFromCloneURL(url) } return Endpoint{} } func (c *Configuration) Remotes() []string { c.loadGitConfig() return c.remotes } func (c *Configuration) GitConfig(key string) (string, bool) { c.loadGitConfig() value, ok := c.gitConfig[strings.ToLower(key)] return value, ok } func (c *Configuration) SetConfig(key, value string) { c.loadGitConfig() c.gitConfig[key] = value } func (c *Configuration) ObjectUrl(oid string) (*url.URL, error) { return ObjectUrl(c.Endpoint(), oid) } func (c *Configuration) loadGitConfig() { c.loading.Lock() defer c.loading.Unlock() if c.gitConfig != nil { return } uniqRemotes := make(map[string]bool) c.gitConfig = make(map[string]string) var output string listOutput, err := git.Config.List() if err != nil { panic(fmt.Errorf("Error listing git config: %s", err)) } configFile := filepath.Join(LocalWorkingDir, ".gitconfig") fileOutput, err := git.Config.ListFromFile(configFile) if err != nil { panic(fmt.Errorf("Error listing git config from file: %s", err)) } localConfig := filepath.Join(LocalGitDir, "config") localOutput, err := git.Config.ListFromFile(localConfig) if err != nil { panic(fmt.Errorf("Error listing git config from file %s", err)) } output = fileOutput + "\n" + listOutput + "\n" + localOutput lines := strings.Split(output, "\n") for _, line := range lines { pieces := strings.SplitN(line, "=", 2) if len(pieces) < 2 { continue } key := strings.ToLower(pieces[0]) c.gitConfig[key] = pieces[1] keyParts := strings.Split(key, ".") if len(keyParts) > 1 && keyParts[0] == "remote" { remote := keyParts[1] uniqRemotes[remote] = remote == "origin" } } c.remotes = make([]string, 0, len(uniqRemotes)) for remote, isOrigin := range uniqRemotes { if isOrigin { continue } c.remotes = append(c.remotes, remote) } }