git-lfs/lfs/config.go

262 lines
5.5 KiB
Go
Raw Normal View History

2015-03-19 19:30:55 +00:00
package lfs
2013-11-05 17:07:03 +00:00
2013-11-05 17:45:01 +00:00
import (
2014-05-22 23:02:56 +00:00
"fmt"
2015-01-12 00:53:10 +00:00
"net/http"
"net/url"
"os"
"path"
"path/filepath"
2014-03-19 21:21:52 +00:00
"regexp"
"strconv"
"strings"
2015-04-23 01:07:52 +00:00
"sync"
2015-05-13 19:43:41 +00:00
"github.com/github/git-lfs/git"
2013-11-05 17:45:01 +00:00
)
2013-11-05 17:07:03 +00:00
type Configuration struct {
2015-02-17 16:39:15 +00:00
CurrentRemote string
httpClient *HttpClient
redirectingHttpClient *http.Client
envVars map[string]string
2015-03-05 19:49:15 +00:00
isTracingHttp bool
isLoggingStats bool
loading sync.Mutex // guards initialization of gitConfig and remotes
gitConfig map[string]string
remotes []string
2013-11-05 17:07:03 +00:00
}
type Endpoint struct {
Url string
SshUserAndHost string
SshPath string
}
2014-05-22 23:02:56 +00:00
var (
2015-03-05 19:59:52 +00:00
Config = NewConfig()
httpPrefixRe = regexp.MustCompile("\\Ahttps?://")
defaultRemote = "origin"
2014-05-22 23:02:56 +00:00
)
2013-11-05 17:07:03 +00:00
2015-03-05 19:59:52 +00:00
func NewConfig() *Configuration {
c := &Configuration{
CurrentRemote: defaultRemote,
envVars: make(map[string]string),
2015-03-05 19:59:52 +00:00
}
c.isTracingHttp = c.GetenvBool("GIT_CURL_VERBOSE", false)
c.isLoggingStats = c.GetenvBool("GIT_LOG_STATS", false)
2015-03-05 19:59:52 +00:00
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
}
// 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 {
2015-03-19 19:30:55 +00:00
if url, ok := c.GitConfig("lfs.url"); ok {
return NewEndpoint(url)
2013-11-05 17:07:03 +00:00
}
if len(c.CurrentRemote) > 0 && c.CurrentRemote != defaultRemote {
if endpoint := c.RemoteEndpoint(c.CurrentRemote); len(endpoint.Url) > 0 {
return endpoint
}
}
return c.RemoteEndpoint(defaultRemote)
2013-11-05 17:07:03 +00:00
}
2013-11-05 17:45:01 +00:00
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
}
2015-05-07 18:37:30 +00:00
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
}
2015-05-07 18:37:30 +00:00
}
return false
}
func (c *Configuration) PrivateAccess() bool {
if v, ok := c.GitConfig("lfs.access"); ok {
if v == "private" || v == "PRIVATE" {
return true
}
}
return false
}
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)
2015-06-13 22:55:23 +00:00
}
2014-03-19 21:21:52 +00:00
2015-06-13 22:55:23 +00:00
return Endpoint{}
}
2014-03-19 21:21:52 +00:00
const ENDPOINT_URL_UNKNOWN = "<unknown>"
// Create a new endpoint from a URL associated with a git clone URL
// The difference to NewEndpoint is that it appends [.git]/info/lfs to the URL since it
// is the clone URL
func NewEndpointFromCloneURL(url string) Endpoint {
e := NewEndpoint(url)
if e.Url != ENDPOINT_URL_UNKNOWN {
// When using main remote URL for HTTP, append info/lfs
if path.Ext(url) == ".git" {
e.Url += "/info/lfs"
} else {
e.Url += ".git/info/lfs"
}
}
return e
}
// Create a new endpoint from a general URL
func NewEndpoint(url string) Endpoint {
2015-06-13 22:55:23 +00:00
e := Endpoint{Url: url}
if !httpPrefixRe.MatchString(url) {
pieces := strings.SplitN(url, ":", 2)
hostPieces := strings.SplitN(pieces[0], "@", 2)
if len(hostPieces) == 2 {
e.SshUserAndHost = pieces[0]
e.SshPath = pieces[1]
e.Url = fmt.Sprintf("https://%s/%s", hostPieces[1], pieces[1])
}
2015-06-13 22:55:23 +00:00
}
return e
}
func (c *Configuration) Remotes() []string {
2015-03-05 19:49:15 +00:00
c.loadGitConfig()
return c.remotes
}
func (c *Configuration) GitConfig(key string) (string, bool) {
2015-03-05 19:49:15 +00:00
c.loadGitConfig()
2015-01-12 00:53:10 +00:00
value, ok := c.gitConfig[strings.ToLower(key)]
return value, ok
}
func (c *Configuration) SetConfig(key, value string) {
2015-03-05 19:49:15 +00:00
c.loadGitConfig()
c.gitConfig[key] = value
}
2015-03-19 20:39:44 +00:00
func (c *Configuration) ObjectUrl(oid string) (*url.URL, error) {
return ObjectUrl(c.Endpoint(), oid)
}
2015-06-13 22:54:16 +00:00
func ObjectUrl(endpoint Endpoint, oid string) (*url.URL, error) {
u, err := url.Parse(endpoint.Url)
if err != nil {
return nil, err
}
2015-06-13 22:54:16 +00:00
u.Path = path.Join(u.Path, "objects")
if len(oid) > 0 {
u.Path = path.Join(u.Path, oid)
}
2015-06-13 22:54:16 +00:00
return u, nil
}
func (c *Configuration) loadGitConfig() {
2015-04-23 01:07:52 +00:00
c.loading.Lock()
defer c.loading.Unlock()
2015-03-05 19:49:15 +00:00
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 {
2014-06-05 18:48:23 +00:00
panic(fmt.Errorf("Error listing git config: %s", err))
}
configFile := filepath.Join(LocalWorkingDir, ".gitconfig")
fileOutput, err := git.Config.ListFromFile(configFile)
if err != nil {
2014-06-05 18:48:23 +00:00
panic(fmt.Errorf("Error listing git config from file: %s", err))
}
output = fileOutput + "\n" + listOutput
lines := strings.Split(output, "\n")
for _, line := range lines {
pieces := strings.SplitN(line, "=", 2)
if len(pieces) < 2 {
continue
}
2015-01-12 00:53:10 +00:00
key := strings.ToLower(pieces[0])
c.gitConfig[key] = pieces[1]
keyParts := strings.Split(key, ".")
if len(keyParts) > 1 && keyParts[0] == "remote" {
2014-02-01 21:04:40 +00:00
remote := keyParts[1]
uniqRemotes[remote] = remote == "origin"
}
}
2014-02-01 21:04:40 +00:00
c.remotes = make([]string, 0, len(uniqRemotes))
for remote, isOrigin := range uniqRemotes {
if isOrigin {
continue
}
c.remotes = append(c.remotes, remote)
}
}