mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-27 00:23:41 +09:00 
			
		
		
		
	Make git clone URL could use current signed-in user (#33091)
close #33086 * Add a special value for "SSH_USER" setting: `(DOER_USERNAME)` * Improve parseRepositoryURL and add tests (now it doesn't have hard dependency on some setting values) Many changes are just adding "ctx" and "doer" argument to functions. By the way, improve app.example.ini, remove all `%(key)s` syntax, it only makes messy and no user really cares about it. Document: https://gitea.com/gitea/docs/pulls/138
This commit is contained in:
		| @@ -78,8 +78,9 @@ RUN_USER = ; git | |||||||
| ;; Set the domain for the server | ;; Set the domain for the server | ||||||
| ;DOMAIN = localhost | ;DOMAIN = localhost | ||||||
| ;; | ;; | ||||||
| ;; Overwrite the automatically generated public URL. Necessary for proxies and docker. | ;; The AppURL used by Gitea to generate absolute links, defaults to "{PROTOCOL}://{DOMAIN}:{HTTP_PORT}/". | ||||||
| ;ROOT_URL = %(PROTOCOL)s://%(DOMAIN)s:%(HTTP_PORT)s/ | ;; Most users should set it to the real website URL of their Gitea instance. | ||||||
|  | ;ROOT_URL = | ||||||
| ;; | ;; | ||||||
| ;; For development purpose only. It makes Gitea handle sub-path ("/sub-path/owner/repo/...") directly when debugging without a reverse proxy. | ;; For development purpose only. It makes Gitea handle sub-path ("/sub-path/owner/repo/...") directly when debugging without a reverse proxy. | ||||||
| ;; DO NOT USE IT IN PRODUCTION!!! | ;; DO NOT USE IT IN PRODUCTION!!! | ||||||
| @@ -103,8 +104,8 @@ RUN_USER = ; git | |||||||
| ;REDIRECT_OTHER_PORT = false | ;REDIRECT_OTHER_PORT = false | ||||||
| ;PORT_TO_REDIRECT = 80 | ;PORT_TO_REDIRECT = 80 | ||||||
| ;; | ;; | ||||||
| ;; expect PROXY protocol header on connections to https redirector. | ;; expect PROXY protocol header on connections to https redirector, defaults to USE_PROXY_PROTOCOL | ||||||
| ;REDIRECTOR_USE_PROXY_PROTOCOL = %(USE_PROXY_PROTOCOL)s | ;REDIRECTOR_USE_PROXY_PROTOCOL = | ||||||
| ;; Minimum and maximum supported TLS versions | ;; Minimum and maximum supported TLS versions | ||||||
| ;SSL_MIN_VERSION=TLSv1.2 | ;SSL_MIN_VERSION=TLSv1.2 | ||||||
| ;SSL_MAX_VERSION= | ;SSL_MAX_VERSION= | ||||||
| @@ -128,13 +129,14 @@ RUN_USER = ; git | |||||||
| ;; most cases you do not need to change the default value. Alter it only if | ;; most cases you do not need to change the default value. Alter it only if | ||||||
| ;; your SSH server node is not the same as HTTP node. For different protocol, the default | ;; your SSH server node is not the same as HTTP node. For different protocol, the default | ||||||
| ;; values are different. If `PROTOCOL` is `http+unix`, the default value is `http://unix/`. | ;; values are different. If `PROTOCOL` is `http+unix`, the default value is `http://unix/`. | ||||||
| ;; If `PROTOCOL` is `fcgi` or `fcgi+unix`, the default value is `%(PROTOCOL)s://%(HTTP_ADDR)s:%(HTTP_PORT)s/`. | ;; If `PROTOCOL` is `fcgi` or `fcgi+unix`, the default value is `{PROTOCOL}://{HTTP_ADDR}:{HTTP_PORT}/`. | ||||||
| ;; If listen on `0.0.0.0`, the default value is `%(PROTOCOL)s://localhost:%(HTTP_PORT)s/`, Otherwise the default | ;; If listen on `0.0.0.0`, the default value is `{PROTOCOL}://localhost:{HTTP_PORT}/`. | ||||||
| ;; value is `%(PROTOCOL)s://%(HTTP_ADDR)s:%(HTTP_PORT)s/`. | ;; Otherwise the default value is `{PROTOCOL}://{HTTP_ADDR}:{HTTP_PORT}/`. | ||||||
| ;LOCAL_ROOT_URL = %(PROTOCOL)s://%(HTTP_ADDR)s:%(HTTP_PORT)s/ | ;; Most users don't need (and shouldn't) set this value. | ||||||
|  | ;LOCAL_ROOT_URL = | ||||||
| ;; | ;; | ||||||
| ;; When making local connections pass the PROXY protocol header. | ;; When making local connections pass the PROXY protocol header, defaults to USE_PROXY_PROTOCOL | ||||||
| ;LOCAL_USE_PROXY_PROTOCOL = %(USE_PROXY_PROTOCOL)s | ;LOCAL_USE_PROXY_PROTOCOL = | ||||||
| ;; | ;; | ||||||
| ;; Disable SSH feature when not available | ;; Disable SSH feature when not available | ||||||
| ;DISABLE_SSH = false | ;DISABLE_SSH = false | ||||||
| @@ -146,13 +148,17 @@ RUN_USER = ; git | |||||||
| ;SSH_SERVER_USE_PROXY_PROTOCOL = false | ;SSH_SERVER_USE_PROXY_PROTOCOL = false | ||||||
| ;; | ;; | ||||||
| ;; Username to use for the builtin SSH server. If blank, then it is the value of RUN_USER. | ;; Username to use for the builtin SSH server. If blank, then it is the value of RUN_USER. | ||||||
| ;BUILTIN_SSH_SERVER_USER = %(RUN_USER)s | ;BUILTIN_SSH_SERVER_USER = | ||||||
| ;; | ;; | ||||||
| ;; Domain name to be exposed in clone URL | ;; Domain name to be exposed in clone URL, defaults to DOMAIN or the domain part of ROOT_URL | ||||||
| ;SSH_DOMAIN = %(DOMAIN)s | ;SSH_DOMAIN = | ||||||
| ;; | ;; | ||||||
| ;; SSH username displayed in clone URLs. | ;; SSH username displayed in clone URLs. It defaults to BUILTIN_SSH_SERVER_USER or RUN_USER. | ||||||
| ;SSH_USER = %(BUILTIN_SSH_SERVER_USER)s | ;; If it is set to "(DOER_USERNAME)", it will use current signed-in user's username. | ||||||
|  | ;; This option is only for some advanced users who have configured their SSH reverse-proxy | ||||||
|  | ;; and need to use different usernames for git SSH clone. | ||||||
|  | ;; Most users should just leave it blank. | ||||||
|  | ;SSH_USER = | ||||||
| ;; | ;; | ||||||
| ;; The network interface the builtin SSH server should listen on | ;; The network interface the builtin SSH server should listen on | ||||||
| ;SSH_LISTEN_HOST = | ;SSH_LISTEN_HOST = | ||||||
| @@ -160,8 +166,8 @@ RUN_USER = ; git | |||||||
| ;; Port number to be exposed in clone URL | ;; Port number to be exposed in clone URL | ||||||
| ;SSH_PORT = 22 | ;SSH_PORT = 22 | ||||||
| ;; | ;; | ||||||
| ;; The port number the builtin SSH server should listen on | ;; The port number the builtin SSH server should listen on, defaults to SSH_PORT | ||||||
| ;SSH_LISTEN_PORT = %(SSH_PORT)s | ;SSH_LISTEN_PORT = | ||||||
| ;; | ;; | ||||||
| ;; Root path of SSH directory, default is '~/.ssh', but you have to use '/home/git/.ssh'. | ;; Root path of SSH directory, default is '~/.ssh', but you have to use '/home/git/.ssh'. | ||||||
| ;SSH_ROOT_PATH = | ;SSH_ROOT_PATH = | ||||||
| @@ -188,7 +194,7 @@ RUN_USER = ; git | |||||||
| ;; | ;; | ||||||
| ;; For the built-in SSH server, choose the keypair to offer as the host key | ;; For the built-in SSH server, choose the keypair to offer as the host key | ||||||
| ;; The private key should be at SSH_SERVER_HOST_KEY and the public SSH_SERVER_HOST_KEY.pub | ;; The private key should be at SSH_SERVER_HOST_KEY and the public SSH_SERVER_HOST_KEY.pub | ||||||
| ;; relative paths are made absolute relative to the %(APP_DATA_PATH)s | ;; relative paths are made absolute relative to the APP_DATA_PATH | ||||||
| ;SSH_SERVER_HOST_KEYS=ssh/gitea.rsa, ssh/gogs.rsa | ;SSH_SERVER_HOST_KEYS=ssh/gitea.rsa, ssh/gogs.rsa | ||||||
| ;; | ;; | ||||||
| ;; Directory to create temporary files in when testing public keys using ssh-keygen, | ;; Directory to create temporary files in when testing public keys using ssh-keygen, | ||||||
| @@ -582,7 +588,7 @@ ENABLED = true | |||||||
| [log] | [log] | ||||||
| ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ||||||
| ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ||||||
| ;; Root path for the log files - defaults to %(GITEA_WORK_DIR)/log | ;; Root path for the log files - defaults to "{AppWorkPath}/log" | ||||||
| ;ROOT_PATH = | ;ROOT_PATH = | ||||||
| ;; | ;; | ||||||
| ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ||||||
| @@ -682,8 +688,8 @@ LEVEL = Info | |||||||
| ;; The path of git executable. If empty, Gitea searches through the PATH environment. | ;; The path of git executable. If empty, Gitea searches through the PATH environment. | ||||||
| ;PATH = | ;PATH = | ||||||
| ;; | ;; | ||||||
| ;; The HOME directory for Git | ;; The HOME directory for Git, defaults to "{APP_DATA_PATH}/home" | ||||||
| ;HOME_PATH = %(APP_DATA_PATH)s/home | ;HOME_PATH = | ||||||
| ;; | ;; | ||||||
| ;; Disables highlight of added and removed changes | ;; Disables highlight of added and removed changes | ||||||
| ;DISABLE_DIFF_HIGHLIGHT = false | ;DISABLE_DIFF_HIGHLIGHT = false | ||||||
| @@ -946,8 +952,8 @@ LEVEL = Info | |||||||
| ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ||||||
| ;[repository] | ;[repository] | ||||||
| ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; | ||||||
| ;; Root path for storing all repository data. By default, it is set to %(APP_DATA_PATH)s/gitea-repositories. | ;; Root path for storing all repository data. By default, it is set to "{APP_DATA_PATH}/gitea-repositories". | ||||||
| ;; A relative path is interpreted as _`AppWorkPath`_/%(ROOT)s | ;; A relative path is interpreted as "{AppWorkPath}/{ROOT}" (use AppWorkPath as base path). | ||||||
| ;ROOT = | ;ROOT = | ||||||
| ;; | ;; | ||||||
| ;; The script type this server supports. Usually this is `bash`, but some users report that only `sh` is available. | ;; The script type this server supports. Usually this is `bash`, but some users report that only `sh` is available. | ||||||
| @@ -1506,7 +1512,8 @@ LEVEL = Info | |||||||
| ;TYPE = persistable-channel | ;TYPE = persistable-channel | ||||||
| ;; | ;; | ||||||
| ;; data-dir for storing persistable queues and level queues, individual queues will default to `queues/common` meaning the queue is shared. | ;; data-dir for storing persistable queues and level queues, individual queues will default to `queues/common` meaning the queue is shared. | ||||||
| ;DATADIR = queues/ ; Relative paths will be made absolute against `%(APP_DATA_PATH)s`. | ;; Relative paths will be made absolute against "APP_DATA_PATH" | ||||||
|  | ;DATADIR = queues/ | ||||||
| ;; | ;; | ||||||
| ;; Default queue length before a channel queue will block | ;; Default queue length before a channel queue will block | ||||||
| ;LENGTH = 100000 | ;LENGTH = 100000 | ||||||
|   | |||||||
| @@ -172,7 +172,7 @@ func getRemoteAddress(ownerName, repoName, remoteName string) (string, error) { | |||||||
| 		return "", fmt.Errorf("get remote %s's address of %s/%s failed: %v", remoteName, ownerName, repoName, err) | 		return "", fmt.Errorf("get remote %s's address of %s/%s failed: %v", remoteName, ownerName, repoName, err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	u, err := giturl.Parse(remoteURL) | 	u, err := giturl.ParseGitURL(remoteURL) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return "", err | 		return "", err | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -20,6 +20,7 @@ import ( | |||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
| 	"code.gitea.io/gitea/modules/base" | 	"code.gitea.io/gitea/modules/base" | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
|  | 	giturl "code.gitea.io/gitea/modules/git/url" | ||||||
| 	"code.gitea.io/gitea/modules/httplib" | 	"code.gitea.io/gitea/modules/httplib" | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/markup" | 	"code.gitea.io/gitea/modules/markup" | ||||||
| @@ -637,14 +638,26 @@ type CloneLink struct { | |||||||
| } | } | ||||||
|  |  | ||||||
| // ComposeHTTPSCloneURL returns HTTPS clone URL based on given owner and repository name. | // ComposeHTTPSCloneURL returns HTTPS clone URL based on given owner and repository name. | ||||||
| func ComposeHTTPSCloneURL(owner, repo string) string { | func ComposeHTTPSCloneURL(ctx context.Context, owner, repo string) string { | ||||||
| 	return fmt.Sprintf("%s%s/%s.git", setting.AppURL, url.PathEscape(owner), url.PathEscape(repo)) | 	return fmt.Sprintf("%s%s/%s.git", httplib.GuessCurrentAppURL(ctx), url.PathEscape(owner), url.PathEscape(repo)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func ComposeSSHCloneURL(ownerName, repoName string) string { | func ComposeSSHCloneURL(doer *user_model.User, ownerName, repoName string) string { | ||||||
| 	sshUser := setting.SSH.User | 	sshUser := setting.SSH.User | ||||||
| 	sshDomain := setting.SSH.Domain | 	sshDomain := setting.SSH.Domain | ||||||
|  |  | ||||||
|  | 	if sshUser == "(DOER_USERNAME)" { | ||||||
|  | 		// Some users use SSH reverse-proxy and need to use the current signed-in username as the SSH user | ||||||
|  | 		// to make the SSH reverse-proxy could prepare the user's public keys ahead. | ||||||
|  | 		// For most cases we have the correct "doer", then use it as the SSH user. | ||||||
|  | 		// If we can't get the doer, then use the built-in SSH user. | ||||||
|  | 		if doer != nil { | ||||||
|  | 			sshUser = doer.Name | ||||||
|  | 		} else { | ||||||
|  | 			sshUser = setting.SSH.BuiltinServerUser | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	// non-standard port, it must use full URI | 	// non-standard port, it must use full URI | ||||||
| 	if setting.SSH.Port != 22 { | 	if setting.SSH.Port != 22 { | ||||||
| 		sshHost := net.JoinHostPort(sshDomain, strconv.Itoa(setting.SSH.Port)) | 		sshHost := net.JoinHostPort(sshDomain, strconv.Itoa(setting.SSH.Port)) | ||||||
| @@ -662,21 +675,20 @@ func ComposeSSHCloneURL(ownerName, repoName string) string { | |||||||
| 	return fmt.Sprintf("%s@%s:%s/%s.git", sshUser, sshHost, url.PathEscape(ownerName), url.PathEscape(repoName)) | 	return fmt.Sprintf("%s@%s:%s/%s.git", sshUser, sshHost, url.PathEscape(ownerName), url.PathEscape(repoName)) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (repo *Repository) cloneLink(isWiki bool) *CloneLink { | func (repo *Repository) cloneLink(ctx context.Context, doer *user_model.User, repoPathName string) *CloneLink { | ||||||
| 	repoName := repo.Name |  | ||||||
| 	if isWiki { |  | ||||||
| 		repoName += ".wiki" |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	cl := new(CloneLink) | 	cl := new(CloneLink) | ||||||
| 	cl.SSH = ComposeSSHCloneURL(repo.OwnerName, repoName) | 	cl.SSH = ComposeSSHCloneURL(doer, repo.OwnerName, repoPathName) | ||||||
| 	cl.HTTPS = ComposeHTTPSCloneURL(repo.OwnerName, repoName) | 	cl.HTTPS = ComposeHTTPSCloneURL(ctx, repo.OwnerName, repoPathName) | ||||||
| 	return cl | 	return cl | ||||||
| } | } | ||||||
|  |  | ||||||
| // CloneLink returns clone URLs of repository. | // CloneLink returns clone URLs of repository. | ||||||
| func (repo *Repository) CloneLink() (cl *CloneLink) { | func (repo *Repository) CloneLink(ctx context.Context, doer *user_model.User) (cl *CloneLink) { | ||||||
| 	return repo.cloneLink(false) | 	return repo.cloneLink(ctx, doer, repo.Name) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (repo *Repository) CloneLinkGeneral(ctx context.Context) (cl *CloneLink) { | ||||||
|  | 	return repo.cloneLink(ctx, nil /* no doer, use a general git user */, repo.Name) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetOriginalURLHostname returns the hostname of a URL or the URL | // GetOriginalURLHostname returns the hostname of a URL or the URL | ||||||
| @@ -772,47 +784,75 @@ func GetRepositoryByName(ctx context.Context, ownerID int64, name string) (*Repo | |||||||
| 	return &repo, err | 	return &repo, err | ||||||
| } | } | ||||||
|  |  | ||||||
| // getRepositoryURLPathSegments returns segments (owner, reponame) extracted from a url | func parseRepositoryURL(ctx context.Context, repoURL string) (ret struct { | ||||||
| func getRepositoryURLPathSegments(repoURL string) []string { | 	OwnerName, RepoName, RemainingPath string | ||||||
| 	if strings.HasPrefix(repoURL, setting.AppURL) { | }, | ||||||
| 		return strings.Split(strings.TrimPrefix(repoURL, setting.AppURL), "/") | ) { | ||||||
| 	} | 	// possible urls for git: | ||||||
|  | 	//  https://my.domain/sub-path/<owner>/<repo>[.git] | ||||||
|  | 	//  git+ssh://user@my.domain/<owner>/<repo>[.git] | ||||||
|  | 	//  ssh://user@my.domain/<owner>/<repo>[.git] | ||||||
|  | 	//  user@my.domain:<owner>/<repo>[.git] | ||||||
|  |  | ||||||
| 	sshURLVariants := [4]string{ | 	fillPathParts := func(s string) { | ||||||
| 		setting.SSH.Domain + ":", | 		s = strings.TrimPrefix(s, "/") | ||||||
| 		setting.SSH.User + "@" + setting.SSH.Domain + ":", | 		fields := strings.SplitN(s, "/", 3) | ||||||
| 		"git+ssh://" + setting.SSH.Domain + "/", | 		if len(fields) >= 2 { | ||||||
| 		"git+ssh://" + setting.SSH.User + "@" + setting.SSH.Domain + "/", | 			ret.OwnerName = fields[0] | ||||||
|  | 			ret.RepoName = strings.TrimSuffix(fields[1], ".git") | ||||||
|  | 			if len(fields) == 3 { | ||||||
|  | 				ret.RemainingPath = "/" + fields[2] | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 	for _, sshURL := range sshURLVariants { |  | ||||||
| 		if strings.HasPrefix(repoURL, sshURL) { |  | ||||||
| 			return strings.Split(strings.TrimPrefix(repoURL, sshURL), "/") |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	parsed, err := giturl.ParseGitURL(repoURL) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return ret | ||||||
|  | 	} | ||||||
|  | 	if parsed.URL.Scheme == "http" || parsed.URL.Scheme == "https" { | ||||||
|  | 		if !httplib.IsCurrentGiteaSiteURL(ctx, repoURL) { | ||||||
|  | 			return ret | ||||||
|  | 		} | ||||||
|  | 		fillPathParts(strings.TrimPrefix(parsed.URL.Path, setting.AppSubURL)) | ||||||
|  | 	} else if parsed.URL.Scheme == "ssh" || parsed.URL.Scheme == "git+ssh" { | ||||||
|  | 		domainSSH := setting.SSH.Domain | ||||||
|  | 		domainCur := httplib.GuessCurrentHostDomain(ctx) | ||||||
|  | 		urlDomain, _, _ := net.SplitHostPort(parsed.URL.Host) | ||||||
|  | 		urlDomain = util.IfZero(urlDomain, parsed.URL.Host) | ||||||
|  | 		if urlDomain == "" { | ||||||
|  | 			return ret | ||||||
|  | 		} | ||||||
|  | 		// check whether URL domain is the App domain | ||||||
|  | 		domainMatches := domainSSH == urlDomain | ||||||
|  | 		// check whether URL domain is current domain from context | ||||||
|  | 		domainMatches = domainMatches || (domainCur != "" && domainCur == urlDomain) | ||||||
|  | 		if domainMatches { | ||||||
|  | 			fillPathParts(parsed.URL.Path) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return ret | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetRepositoryByURL returns the repository by given url | // GetRepositoryByURL returns the repository by given url | ||||||
| func GetRepositoryByURL(ctx context.Context, repoURL string) (*Repository, error) { | func GetRepositoryByURL(ctx context.Context, repoURL string) (*Repository, error) { | ||||||
| 	// possible urls for git: | 	ret := parseRepositoryURL(ctx, repoURL) | ||||||
| 	//  https://my.domain/sub-path/<owner>/<repo>.git | 	if ret.OwnerName == "" { | ||||||
| 	//  https://my.domain/sub-path/<owner>/<repo> |  | ||||||
| 	//  git+ssh://user@my.domain/<owner>/<repo>.git |  | ||||||
| 	//  git+ssh://user@my.domain/<owner>/<repo> |  | ||||||
| 	//  user@my.domain:<owner>/<repo>.git |  | ||||||
| 	//  user@my.domain:<owner>/<repo> |  | ||||||
|  |  | ||||||
| 	pathSegments := getRepositoryURLPathSegments(repoURL) |  | ||||||
|  |  | ||||||
| 	if len(pathSegments) != 2 { |  | ||||||
| 		return nil, fmt.Errorf("unknown or malformed repository URL") | 		return nil, fmt.Errorf("unknown or malformed repository URL") | ||||||
| 	} | 	} | ||||||
|  | 	return GetRepositoryByOwnerAndName(ctx, ret.OwnerName, ret.RepoName) | ||||||
|  | } | ||||||
|  |  | ||||||
| 	ownerName := pathSegments[0] | // GetRepositoryByURLRelax also accepts an SSH clone URL without user part | ||||||
| 	repoName := strings.TrimSuffix(pathSegments[1], ".git") | func GetRepositoryByURLRelax(ctx context.Context, repoURL string) (*Repository, error) { | ||||||
| 	return GetRepositoryByOwnerAndName(ctx, ownerName, repoName) | 	if !strings.Contains(repoURL, "://") && !strings.Contains(repoURL, "@") { | ||||||
|  | 		// convert "example.com:owner/repo" to "@example.com:owner/repo" | ||||||
|  | 		p1, p2, p3 := strings.Index(repoURL, "."), strings.Index(repoURL, ":"), strings.Index(repoURL, "/") | ||||||
|  | 		if 0 < p1 && p1 < p2 && p2 < p3 { | ||||||
|  | 			repoURL = "@" + repoURL | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return GetRepositoryByURL(ctx, repoURL) | ||||||
| } | } | ||||||
|  |  | ||||||
| // GetRepositoryByID returns the repository by given id if exists. | // GetRepositoryByID returns the repository by given id if exists. | ||||||
|   | |||||||
| @@ -4,18 +4,23 @@ | |||||||
| package repo | package repo | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"net/http" | ||||||
|  | 	"net/url" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| 	"code.gitea.io/gitea/models/db" | 	"code.gitea.io/gitea/models/db" | ||||||
| 	"code.gitea.io/gitea/models/unit" | 	"code.gitea.io/gitea/models/unit" | ||||||
| 	"code.gitea.io/gitea/models/unittest" | 	"code.gitea.io/gitea/models/unittest" | ||||||
| 	user_model "code.gitea.io/gitea/models/user" | 	user_model "code.gitea.io/gitea/models/user" | ||||||
|  | 	"code.gitea.io/gitea/modules/httplib" | ||||||
| 	"code.gitea.io/gitea/modules/markup" | 	"code.gitea.io/gitea/modules/markup" | ||||||
| 	"code.gitea.io/gitea/modules/optional" | 	"code.gitea.io/gitea/modules/optional" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	"code.gitea.io/gitea/modules/test" | 	"code.gitea.io/gitea/modules/test" | ||||||
|  |  | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
|  | 	"github.com/stretchr/testify/require" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| @@ -127,65 +132,121 @@ func TestMetas(t *testing.T) { | |||||||
| 	assert.Equal(t, ",owners,team1,", metas["teams"]) | 	assert.Equal(t, ",owners,team1,", metas["teams"]) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestParseRepositoryURLPathSegments(t *testing.T) { | ||||||
|  | 	defer test.MockVariableValue(&setting.AppURL, "https://localhost:3000")() | ||||||
|  |  | ||||||
|  | 	ctxURL, _ := url.Parse("https://gitea") | ||||||
|  | 	ctxReq := &http.Request{URL: ctxURL, Header: http.Header{}} | ||||||
|  | 	ctxReq.Host = ctxURL.Host | ||||||
|  | 	ctxReq.Header.Add("X-Forwarded-Proto", ctxURL.Scheme) | ||||||
|  | 	ctx := context.WithValue(context.Background(), httplib.RequestContextKey, ctxReq) | ||||||
|  | 	cases := []struct { | ||||||
|  | 		input                          string | ||||||
|  | 		ownerName, repoName, remaining string | ||||||
|  | 	}{ | ||||||
|  | 		{input: "/user/repo"}, | ||||||
|  |  | ||||||
|  | 		{input: "https://localhost:3000/user/repo", ownerName: "user", repoName: "repo"}, | ||||||
|  | 		{input: "https://external:3000/user/repo"}, | ||||||
|  |  | ||||||
|  | 		{input: "https://localhost:3000/user/repo.git/other", ownerName: "user", repoName: "repo", remaining: "/other"}, | ||||||
|  |  | ||||||
|  | 		{input: "https://gitea/user/repo", ownerName: "user", repoName: "repo"}, | ||||||
|  | 		{input: "https://gitea:3333/user/repo"}, | ||||||
|  |  | ||||||
|  | 		{input: "ssh://try.gitea.io:2222/user/repo", ownerName: "user", repoName: "repo"}, | ||||||
|  | 		{input: "ssh://external:2222/user/repo"}, | ||||||
|  |  | ||||||
|  | 		{input: "git+ssh://user@try.gitea.io/user/repo.git", ownerName: "user", repoName: "repo"}, | ||||||
|  | 		{input: "git+ssh://user@external/user/repo.git"}, | ||||||
|  |  | ||||||
|  | 		{input: "root@try.gitea.io:user/repo.git", ownerName: "user", repoName: "repo"}, | ||||||
|  | 		{input: "root@gitea:user/repo.git", ownerName: "user", repoName: "repo"}, | ||||||
|  | 		{input: "root@external:user/repo.git"}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, c := range cases { | ||||||
|  | 		t.Run(c.input, func(t *testing.T) { | ||||||
|  | 			ret := parseRepositoryURL(ctx, c.input) | ||||||
|  | 			assert.Equal(t, c.ownerName, ret.OwnerName) | ||||||
|  | 			assert.Equal(t, c.repoName, ret.RepoName) | ||||||
|  | 			assert.Equal(t, c.remaining, ret.RemainingPath) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	t.Run("WithSubpath", func(t *testing.T) { | ||||||
|  | 		defer test.MockVariableValue(&setting.AppURL, "https://localhost:3000/subpath")() | ||||||
|  | 		defer test.MockVariableValue(&setting.AppSubURL, "/subpath")() | ||||||
|  | 		cases = []struct { | ||||||
|  | 			input                          string | ||||||
|  | 			ownerName, repoName, remaining string | ||||||
|  | 		}{ | ||||||
|  | 			{input: "https://localhost:3000/user/repo"}, | ||||||
|  | 			{input: "https://localhost:3000/subpath/user/repo.git/other", ownerName: "user", repoName: "repo", remaining: "/other"}, | ||||||
|  |  | ||||||
|  | 			{input: "ssh://try.gitea.io:2222/user/repo", ownerName: "user", repoName: "repo"}, | ||||||
|  | 			{input: "ssh://external:2222/user/repo"}, | ||||||
|  |  | ||||||
|  | 			{input: "git+ssh://user@try.gitea.io/user/repo.git", ownerName: "user", repoName: "repo"}, | ||||||
|  | 			{input: "git+ssh://user@external/user/repo.git"}, | ||||||
|  |  | ||||||
|  | 			{input: "root@try.gitea.io:user/repo.git", ownerName: "user", repoName: "repo"}, | ||||||
|  | 			{input: "root@external:user/repo.git"}, | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for _, c := range cases { | ||||||
|  | 			t.Run(c.input, func(t *testing.T) { | ||||||
|  | 				ret := parseRepositoryURL(ctx, c.input) | ||||||
|  | 				assert.Equal(t, c.ownerName, ret.OwnerName) | ||||||
|  | 				assert.Equal(t, c.repoName, ret.RepoName) | ||||||
|  | 				assert.Equal(t, c.remaining, ret.RemainingPath) | ||||||
|  | 			}) | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  |  | ||||||
| func TestGetRepositoryByURL(t *testing.T) { | func TestGetRepositoryByURL(t *testing.T) { | ||||||
| 	assert.NoError(t, unittest.PrepareTestDatabase()) | 	assert.NoError(t, unittest.PrepareTestDatabase()) | ||||||
|  |  | ||||||
| 	t.Run("InvalidPath", func(t *testing.T) { | 	t.Run("InvalidPath", func(t *testing.T) { | ||||||
| 		repo, err := GetRepositoryByURL(db.DefaultContext, "something") | 		repo, err := GetRepositoryByURL(db.DefaultContext, "something") | ||||||
|  |  | ||||||
| 		assert.Nil(t, repo) | 		assert.Nil(t, repo) | ||||||
| 		assert.Error(t, err) | 		assert.Error(t, err) | ||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| 	t.Run("ValidHttpURL", func(t *testing.T) { | 	testRepo2 := func(t *testing.T, url string) { | ||||||
| 		test := func(t *testing.T, url string) { |  | ||||||
| 		repo, err := GetRepositoryByURL(db.DefaultContext, url) | 		repo, err := GetRepositoryByURL(db.DefaultContext, url) | ||||||
|  | 		require.NoError(t, err) | ||||||
| 			assert.NotNil(t, repo) | 		assert.EqualValues(t, 2, repo.ID) | ||||||
| 			assert.NoError(t, err) | 		assert.EqualValues(t, 2, repo.OwnerID) | ||||||
|  |  | ||||||
| 			assert.Equal(t, int64(2), repo.ID) |  | ||||||
| 			assert.Equal(t, int64(2), repo.OwnerID) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 		test(t, "https://try.gitea.io/user2/repo2") | 	t.Run("ValidHttpURL", func(t *testing.T) { | ||||||
| 		test(t, "https://try.gitea.io/user2/repo2.git") | 		testRepo2(t, "https://try.gitea.io/user2/repo2") | ||||||
|  | 		testRepo2(t, "https://try.gitea.io/user2/repo2.git") | ||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| 	t.Run("ValidGitSshURL", func(t *testing.T) { | 	t.Run("ValidGitSshURL", func(t *testing.T) { | ||||||
| 		test := func(t *testing.T, url string) { | 		testRepo2(t, "git+ssh://sshuser@try.gitea.io/user2/repo2") | ||||||
| 			repo, err := GetRepositoryByURL(db.DefaultContext, url) | 		testRepo2(t, "git+ssh://sshuser@try.gitea.io/user2/repo2.git") | ||||||
|  |  | ||||||
| 			assert.NotNil(t, repo) | 		testRepo2(t, "git+ssh://try.gitea.io/user2/repo2") | ||||||
| 			assert.NoError(t, err) | 		testRepo2(t, "git+ssh://try.gitea.io/user2/repo2.git") | ||||||
|  |  | ||||||
| 			assert.Equal(t, int64(2), repo.ID) |  | ||||||
| 			assert.Equal(t, int64(2), repo.OwnerID) |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		test(t, "git+ssh://sshuser@try.gitea.io/user2/repo2") |  | ||||||
| 		test(t, "git+ssh://sshuser@try.gitea.io/user2/repo2.git") |  | ||||||
|  |  | ||||||
| 		test(t, "git+ssh://try.gitea.io/user2/repo2") |  | ||||||
| 		test(t, "git+ssh://try.gitea.io/user2/repo2.git") |  | ||||||
| 	}) | 	}) | ||||||
|  |  | ||||||
| 	t.Run("ValidImplicitSshURL", func(t *testing.T) { | 	t.Run("ValidImplicitSshURL", func(t *testing.T) { | ||||||
| 		test := func(t *testing.T, url string) { | 		testRepo2(t, "sshuser@try.gitea.io:user2/repo2") | ||||||
| 			repo, err := GetRepositoryByURL(db.DefaultContext, url) | 		testRepo2(t, "sshuser@try.gitea.io:user2/repo2.git") | ||||||
|  |  | ||||||
| 			assert.NotNil(t, repo) |  | ||||||
| 			assert.NoError(t, err) |  | ||||||
|  |  | ||||||
|  | 		testRelax := func(t *testing.T, url string) { | ||||||
|  | 			repo, err := GetRepositoryByURLRelax(db.DefaultContext, url) | ||||||
|  | 			require.NoError(t, err) | ||||||
| 			assert.Equal(t, int64(2), repo.ID) | 			assert.Equal(t, int64(2), repo.ID) | ||||||
| 			assert.Equal(t, int64(2), repo.OwnerID) | 			assert.Equal(t, int64(2), repo.OwnerID) | ||||||
| 		} | 		} | ||||||
|  | 		// TODO: it doesn't seem to be common git ssh URL, should we really support this? | ||||||
| 		test(t, "sshuser@try.gitea.io:user2/repo2") | 		testRelax(t, "try.gitea.io:user2/repo2") | ||||||
| 		test(t, "sshuser@try.gitea.io:user2/repo2.git") | 		testRelax(t, "try.gitea.io:user2/repo2.git") | ||||||
|  |  | ||||||
| 		test(t, "try.gitea.io:user2/repo2") |  | ||||||
| 		test(t, "try.gitea.io:user2/repo2.git") |  | ||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -199,23 +260,30 @@ func TestComposeSSHCloneURL(t *testing.T) { | |||||||
| 	setting.SSH.Domain = "domain" | 	setting.SSH.Domain = "domain" | ||||||
| 	setting.SSH.Port = 22 | 	setting.SSH.Port = 22 | ||||||
| 	setting.Repository.UseCompatSSHURI = false | 	setting.Repository.UseCompatSSHURI = false | ||||||
| 	assert.Equal(t, "git@domain:user/repo.git", ComposeSSHCloneURL("user", "repo")) | 	assert.Equal(t, "git@domain:user/repo.git", ComposeSSHCloneURL(&user_model.User{Name: "doer"}, "user", "repo")) | ||||||
| 	setting.Repository.UseCompatSSHURI = true | 	setting.Repository.UseCompatSSHURI = true | ||||||
| 	assert.Equal(t, "ssh://git@domain/user/repo.git", ComposeSSHCloneURL("user", "repo")) | 	assert.Equal(t, "ssh://git@domain/user/repo.git", ComposeSSHCloneURL(&user_model.User{Name: "doer"}, "user", "repo")) | ||||||
| 	// test SSH_DOMAIN while use non-standard SSH port | 	// test SSH_DOMAIN while use non-standard SSH port | ||||||
| 	setting.SSH.Port = 123 | 	setting.SSH.Port = 123 | ||||||
| 	setting.Repository.UseCompatSSHURI = false | 	setting.Repository.UseCompatSSHURI = false | ||||||
| 	assert.Equal(t, "ssh://git@domain:123/user/repo.git", ComposeSSHCloneURL("user", "repo")) | 	assert.Equal(t, "ssh://git@domain:123/user/repo.git", ComposeSSHCloneURL(nil, "user", "repo")) | ||||||
| 	setting.Repository.UseCompatSSHURI = true | 	setting.Repository.UseCompatSSHURI = true | ||||||
| 	assert.Equal(t, "ssh://git@domain:123/user/repo.git", ComposeSSHCloneURL("user", "repo")) | 	assert.Equal(t, "ssh://git@domain:123/user/repo.git", ComposeSSHCloneURL(nil, "user", "repo")) | ||||||
|  |  | ||||||
| 	// test IPv6 SSH_DOMAIN | 	// test IPv6 SSH_DOMAIN | ||||||
| 	setting.Repository.UseCompatSSHURI = false | 	setting.Repository.UseCompatSSHURI = false | ||||||
| 	setting.SSH.Domain = "::1" | 	setting.SSH.Domain = "::1" | ||||||
| 	setting.SSH.Port = 22 | 	setting.SSH.Port = 22 | ||||||
| 	assert.Equal(t, "git@[::1]:user/repo.git", ComposeSSHCloneURL("user", "repo")) | 	assert.Equal(t, "git@[::1]:user/repo.git", ComposeSSHCloneURL(nil, "user", "repo")) | ||||||
| 	setting.SSH.Port = 123 | 	setting.SSH.Port = 123 | ||||||
| 	assert.Equal(t, "ssh://git@[::1]:123/user/repo.git", ComposeSSHCloneURL("user", "repo")) | 	assert.Equal(t, "ssh://git@[::1]:123/user/repo.git", ComposeSSHCloneURL(nil, "user", "repo")) | ||||||
|  |  | ||||||
|  | 	setting.SSH.User = "(DOER_USERNAME)" | ||||||
|  | 	setting.SSH.Domain = "domain" | ||||||
|  | 	setting.SSH.Port = 22 | ||||||
|  | 	assert.Equal(t, "doer@domain:user/repo.git", ComposeSSHCloneURL(&user_model.User{Name: "doer"}, "user", "repo")) | ||||||
|  | 	setting.SSH.Port = 123 | ||||||
|  | 	assert.Equal(t, "ssh://doer@domain:123/user/repo.git", ComposeSSHCloneURL(&user_model.User{Name: "doer"}, "user", "repo")) | ||||||
| } | } | ||||||
|  |  | ||||||
| func TestIsUsableRepoName(t *testing.T) { | func TestIsUsableRepoName(t *testing.T) { | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ | |||||||
| package repo | package repo | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"context" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
| 	"strings" | 	"strings" | ||||||
| @@ -72,8 +73,8 @@ func (err ErrWikiInvalidFileName) Unwrap() error { | |||||||
| } | } | ||||||
|  |  | ||||||
| // WikiCloneLink returns clone URLs of repository wiki. | // WikiCloneLink returns clone URLs of repository wiki. | ||||||
| func (repo *Repository) WikiCloneLink() *CloneLink { | func (repo *Repository) WikiCloneLink(ctx context.Context, doer *user_model.User) *CloneLink { | ||||||
| 	return repo.cloneLink(true) | 	return repo.cloneLink(ctx, doer, repo.Name+".wiki") | ||||||
| } | } | ||||||
|  |  | ||||||
| // WikiPath returns wiki data path by given user and repository name. | // WikiPath returns wiki data path by given user and repository name. | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ | |||||||
| package repo_test | package repo_test | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"context" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
| @@ -18,7 +19,7 @@ func TestRepository_WikiCloneLink(t *testing.T) { | |||||||
| 	assert.NoError(t, unittest.PrepareTestDatabase()) | 	assert.NoError(t, unittest.PrepareTestDatabase()) | ||||||
|  |  | ||||||
| 	repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1}) | 	repo := unittest.AssertExistsAndLoadBean(t, &repo_model.Repository{ID: 1}) | ||||||
| 	cloneLink := repo.WikiCloneLink() | 	cloneLink := repo.WikiCloneLink(context.Background(), nil) | ||||||
| 	assert.Equal(t, "ssh://sshuser@try.gitea.io:3000/user2/repo1.wiki.git", cloneLink.SSH) | 	assert.Equal(t, "ssh://sshuser@try.gitea.io:3000/user2/repo1.wiki.git", cloneLink.SSH) | ||||||
| 	assert.Equal(t, "https://try.gitea.io/user2/repo1.wiki.git", cloneLink.HTTPS) | 	assert.Equal(t, "https://try.gitea.io/user2/repo1.wiki.git", cloneLink.HTTPS) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -84,6 +84,7 @@ func MainTest(m *testing.M, testOptsArg ...*TestOptions) { | |||||||
|  |  | ||||||
| 	setting.IsInTesting = true | 	setting.IsInTesting = true | ||||||
| 	setting.AppURL = "https://try.gitea.io/" | 	setting.AppURL = "https://try.gitea.io/" | ||||||
|  | 	setting.Domain = "try.gitea.io" | ||||||
| 	setting.RunUser = "runuser" | 	setting.RunUser = "runuser" | ||||||
| 	setting.SSH.User = "sshuser" | 	setting.SSH.User = "sshuser" | ||||||
| 	setting.SSH.BuiltinServerUser = "builtinuser" | 	setting.SSH.BuiltinServerUser = "builtinuser" | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ func GetRemoteURL(ctx context.Context, repoPath, remoteName string) (*giturl.Git | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	return giturl.Parse(addr) | 	return giturl.ParseGitURL(addr) | ||||||
| } | } | ||||||
|  |  | ||||||
| // ErrInvalidCloneAddr represents a "InvalidCloneAddr" kind of error. | // ErrInvalidCloneAddr represents a "InvalidCloneAddr" kind of error. | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ func (err ErrWrongURLFormat) Error() string { | |||||||
| // GitURL represents a git URL | // GitURL represents a git URL | ||||||
| type GitURL struct { | type GitURL struct { | ||||||
| 	*stdurl.URL | 	*stdurl.URL | ||||||
| 	extraMark int // 0 no extra 1 scp 2 file path with no prefix | 	extraMark int // 0: standard URL with scheme, 1: scp short syntax (no scheme), 2: file path with no prefix | ||||||
| } | } | ||||||
|  |  | ||||||
| // String returns the URL's string | // String returns the URL's string | ||||||
| @@ -38,8 +38,11 @@ func (u *GitURL) String() string { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| // Parse parse all kinds of git URL | // ParseGitURL parse all kinds of git URL: | ||||||
| func Parse(remote string) (*GitURL, error) { | // * Full URL: http://git@host/path, http://git@host:port/path | ||||||
|  | // * SCP short syntax: git@host:/path | ||||||
|  | // * File path: /dir/repo/path | ||||||
|  | func ParseGitURL(remote string) (*GitURL, error) { | ||||||
| 	if strings.Contains(remote, "://") { | 	if strings.Contains(remote, "://") { | ||||||
| 		u, err := stdurl.Parse(remote) | 		u, err := stdurl.Parse(remote) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
|   | |||||||
| @@ -157,7 +157,7 @@ func TestParseGitURLs(t *testing.T) { | |||||||
|  |  | ||||||
| 	for _, kase := range kases { | 	for _, kase := range kases { | ||||||
| 		t.Run(kase.kase, func(t *testing.T) { | 		t.Run(kase.kase, func(t *testing.T) { | ||||||
| 			u, err := Parse(kase.kase) | 			u, err := ParseGitURL(kase.kase) | ||||||
| 			assert.NoError(t, err) | 			assert.NoError(t, err) | ||||||
| 			assert.EqualValues(t, kase.expected.extraMark, u.extraMark) | 			assert.EqualValues(t, kase.expected.extraMark, u.extraMark) | ||||||
| 			assert.EqualValues(t, *kase.expected, *u) | 			assert.EqualValues(t, *kase.expected, *u) | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ package httplib | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"context" | 	"context" | ||||||
|  | 	"net" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"net/url" | 	"net/url" | ||||||
| 	"strings" | 	"strings" | ||||||
| @@ -81,6 +82,12 @@ func GuessCurrentHostURL(ctx context.Context) string { | |||||||
| 	return reqScheme + "://" + req.Host | 	return reqScheme + "://" + req.Host | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func GuessCurrentHostDomain(ctx context.Context) string { | ||||||
|  | 	_, host, _ := strings.Cut(GuessCurrentHostURL(ctx), "://") | ||||||
|  | 	domain, _, _ := net.SplitHostPort(host) | ||||||
|  | 	return util.IfZero(domain, host) | ||||||
|  | } | ||||||
|  |  | ||||||
| // MakeAbsoluteURL tries to make a link to an absolute URL: | // MakeAbsoluteURL tries to make a link to an absolute URL: | ||||||
| // * If link is empty, it returns the current app URL. | // * If link is empty, it returns the current app URL. | ||||||
| // * If link is absolute, it returns the link. | // * If link is absolute, it returns the link. | ||||||
| @@ -105,7 +112,7 @@ func IsCurrentGiteaSiteURL(ctx context.Context, s string) bool { | |||||||
| 		if cleanedPath == "" || cleanedPath == "." { | 		if cleanedPath == "" || cleanedPath == "." { | ||||||
| 			u.Path = "/" | 			u.Path = "/" | ||||||
| 		} else { | 		} else { | ||||||
| 			u.Path += "/" + cleanedPath + "/" | 			u.Path = "/" + cleanedPath + "/" | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	if urlIsRelative(s, u) { | 	if urlIsRelative(s, u) { | ||||||
|   | |||||||
| @@ -150,7 +150,7 @@ func mirrorRemoteAddress(ctx context.Context, m *repo_model.Repository, remoteNa | |||||||
| 		return ret | 		return ret | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	u, err := giturl.Parse(remoteURL) | 	u, err := giturl.ParseGitURL(remoteURL) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Error("giturl.Parse %v", err) | 		log.Error("giturl.Parse %v", err) | ||||||
| 		return ret | 		return ret | ||||||
|   | |||||||
| @@ -163,7 +163,7 @@ func UploadPackage(ctx *context.Context) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	repo, err := repo_model.GetRepositoryByURL(ctx, npmPackage.Metadata.Repository.URL) | 	repo, err := repo_model.GetRepositoryByURLRelax(ctx, npmPackage.Metadata.Repository.URL) | ||||||
| 	if err == nil { | 	if err == nil { | ||||||
| 		canWrite := repo.OwnerID == ctx.Doer.ID | 		canWrite := repo.OwnerID == ctx.Doer.ID | ||||||
|  |  | ||||||
|   | |||||||
| @@ -69,9 +69,9 @@ func goGet(ctx *context.Context) { | |||||||
|  |  | ||||||
| 	var cloneURL string | 	var cloneURL string | ||||||
| 	if setting.Repository.GoGetCloneURLProtocol == "ssh" { | 	if setting.Repository.GoGetCloneURLProtocol == "ssh" { | ||||||
| 		cloneURL = repo_model.ComposeSSHCloneURL(ownerName, repoName) | 		cloneURL = repo_model.ComposeSSHCloneURL(ctx.Doer, ownerName, repoName) | ||||||
| 	} else { | 	} else { | ||||||
| 		cloneURL = repo_model.ComposeHTTPSCloneURL(ownerName, repoName) | 		cloneURL = repo_model.ComposeHTTPSCloneURL(ctx, ownerName, repoName) | ||||||
| 	} | 	} | ||||||
| 	goImportContent := fmt.Sprintf("%s git %s", goGetImport, cloneURL /*CloneLink*/) | 	goImportContent := fmt.Sprintf("%s git %s", goGetImport, cloneURL /*CloneLink*/) | ||||||
| 	goSourceContent := fmt.Sprintf("%s _ %s %s", goGetImport, prefix+"{/dir}" /*GoDocDirectory*/, prefix+"{/dir}/{file}#L{line}" /*GoDocFile*/) | 	goSourceContent := fmt.Sprintf("%s _ %s %s", goGetImport, prefix+"{/dir}" /*GoDocDirectory*/, prefix+"{/dir}/{file}#L{line}" /*GoDocFile*/) | ||||||
|   | |||||||
| @@ -1450,7 +1450,7 @@ func registerRoutes(m *web.Router) { | |||||||
| 		m.Get("/raw/*", repo.WikiRaw) | 		m.Get("/raw/*", repo.WikiRaw) | ||||||
| 	}, optSignIn, context.RepoAssignment, repo.MustEnableWiki, reqRepoWikiReader, func(ctx *context.Context) { | 	}, optSignIn, context.RepoAssignment, repo.MustEnableWiki, reqRepoWikiReader, func(ctx *context.Context) { | ||||||
| 		ctx.Data["PageIsWiki"] = true | 		ctx.Data["PageIsWiki"] = true | ||||||
| 		ctx.Data["CloneButtonOriginLink"] = ctx.Repo.Repository.WikiCloneLink() | 		ctx.Data["CloneButtonOriginLink"] = ctx.Repo.Repository.WikiCloneLink(ctx, ctx.Doer) | ||||||
| 	}) | 	}) | ||||||
| 	// end "/{username}/{reponame}/wiki" | 	// end "/{username}/{reponame}/wiki" | ||||||
|  |  | ||||||
|   | |||||||
| @@ -325,9 +325,9 @@ func EarlyResponseForGoGetMeta(ctx *Context) { | |||||||
|  |  | ||||||
| 	var cloneURL string | 	var cloneURL string | ||||||
| 	if setting.Repository.GoGetCloneURLProtocol == "ssh" { | 	if setting.Repository.GoGetCloneURLProtocol == "ssh" { | ||||||
| 		cloneURL = repo_model.ComposeSSHCloneURL(username, reponame) | 		cloneURL = repo_model.ComposeSSHCloneURL(ctx.Doer, username, reponame) | ||||||
| 	} else { | 	} else { | ||||||
| 		cloneURL = repo_model.ComposeHTTPSCloneURL(username, reponame) | 		cloneURL = repo_model.ComposeHTTPSCloneURL(ctx, username, reponame) | ||||||
| 	} | 	} | ||||||
| 	goImportContent := fmt.Sprintf("%s git %s", ComposeGoGetImport(ctx, username, reponame), cloneURL) | 	goImportContent := fmt.Sprintf("%s git %s", ComposeGoGetImport(ctx, username, reponame), cloneURL) | ||||||
| 	htmlMeta := fmt.Sprintf(`<meta name="go-import" content="%s">`, html.EscapeString(goImportContent)) | 	htmlMeta := fmt.Sprintf(`<meta name="go-import" content="%s">`, html.EscapeString(goImportContent)) | ||||||
| @@ -564,7 +564,7 @@ func RepoAssignment(ctx *Context) { | |||||||
| 	// If multiple forks are available or if the user can fork to another account, but there is already a fork: open selection dialog | 	// If multiple forks are available or if the user can fork to another account, but there is already a fork: open selection dialog | ||||||
| 	ctx.Data["ShowForkModal"] = len(userAndOrgForks) > 1 || (canSignedUserFork && len(userAndOrgForks) > 0) | 	ctx.Data["ShowForkModal"] = len(userAndOrgForks) > 1 || (canSignedUserFork && len(userAndOrgForks) > 0) | ||||||
|  |  | ||||||
| 	ctx.Data["RepoCloneLink"] = repo.CloneLink() | 	ctx.Data["RepoCloneLink"] = repo.CloneLink(ctx, ctx.Doer) | ||||||
|  |  | ||||||
| 	cloneButtonShowHTTPS := !setting.Repository.DisableHTTPGit | 	cloneButtonShowHTTPS := !setting.Repository.DisableHTTPGit | ||||||
| 	cloneButtonShowSSH := !setting.SSH.Disabled && (ctx.IsSigned || setting.SSH.ExposeAnonymous) | 	cloneButtonShowSSH := !setting.SSH.Disabled && (ctx.IsSigned || setting.SSH.ExposeAnonymous) | ||||||
|   | |||||||
| @@ -33,7 +33,9 @@ func innerToRepo(ctx context.Context, repo *repo_model.Repository, permissionInR | |||||||
| 		permissionInRepo.SetUnitsWithDefaultAccessMode(repo.Units, permissionInRepo.AccessMode) | 		permissionInRepo.SetUnitsWithDefaultAccessMode(repo.Units, permissionInRepo.AccessMode) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	cloneLink := repo.CloneLink() | 	// TODO: ideally we should pass "doer" into "ToRepo" to to make CloneLink could generate user-related links | ||||||
|  | 	// And passing "doer" in will also fix other FIXMEs in this file. | ||||||
|  | 	cloneLink := repo.CloneLinkGeneral(ctx) // no doer at the moment | ||||||
| 	permission := &api.Permission{ | 	permission := &api.Permission{ | ||||||
| 		Admin: permissionInRepo.AccessMode >= perm.AccessModeAdmin, | 		Admin: permissionInRepo.AccessMode >= perm.AccessModeAdmin, | ||||||
| 		Push:  permissionInRepo.UnitAccessMode(unit_model.TypeCode) >= perm.AccessModeWrite, | 		Push:  permissionInRepo.UnitAccessMode(unit_model.TypeCode) >= perm.AccessModeWrite, | ||||||
|   | |||||||
| @@ -32,7 +32,7 @@ const gitShortEmptySha = "0000000" | |||||||
|  |  | ||||||
| // UpdateAddress writes new address to Git repository and database | // UpdateAddress writes new address to Git repository and database | ||||||
| func UpdateAddress(ctx context.Context, m *repo_model.Mirror, addr string) error { | func UpdateAddress(ctx context.Context, m *repo_model.Mirror, addr string) error { | ||||||
| 	u, err := giturl.Parse(addr) | 	u, err := giturl.ParseGitURL(addr) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return fmt.Errorf("invalid addr: %v", err) | 		return fmt.Errorf("invalid addr: %v", err) | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -79,7 +79,7 @@ func prepareRepoCommit(ctx context.Context, repo *repo_model.Repository, tmpDir, | |||||||
| 		return fmt.Errorf("GetRepoInitFile[%s]: %w", opts.Readme, err) | 		return fmt.Errorf("GetRepoInitFile[%s]: %w", opts.Readme, err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	cloneLink := repo.CloneLink() | 	cloneLink := repo.CloneLink(ctx, nil /* no doer so do not generate user-related SSH link */) | ||||||
| 	match := map[string]string{ | 	match := map[string]string{ | ||||||
| 		"Name":           repo.Name, | 		"Name":           repo.Name, | ||||||
| 		"Description":    repo.Description, | 		"Description":    repo.Description, | ||||||
|   | |||||||
| @@ -51,7 +51,7 @@ var defaultTransformers = []transformer{ | |||||||
| 	{Name: "TITLE", Transform: util.ToTitleCase}, | 	{Name: "TITLE", Transform: util.ToTitleCase}, | ||||||
| } | } | ||||||
|  |  | ||||||
| func generateExpansion(src string, templateRepo, generateRepo *repo_model.Repository, sanitizeFileName bool) string { | func generateExpansion(ctx context.Context, src string, templateRepo, generateRepo *repo_model.Repository, sanitizeFileName bool) string { | ||||||
| 	year, month, day := time.Now().Date() | 	year, month, day := time.Now().Date() | ||||||
| 	expansions := []expansion{ | 	expansions := []expansion{ | ||||||
| 		{Name: "YEAR", Value: strconv.Itoa(year), Transformers: nil}, | 		{Name: "YEAR", Value: strconv.Itoa(year), Transformers: nil}, | ||||||
| @@ -66,10 +66,10 @@ func generateExpansion(src string, templateRepo, generateRepo *repo_model.Reposi | |||||||
| 		{Name: "TEMPLATE_OWNER", Value: templateRepo.OwnerName, Transformers: defaultTransformers}, | 		{Name: "TEMPLATE_OWNER", Value: templateRepo.OwnerName, Transformers: defaultTransformers}, | ||||||
| 		{Name: "REPO_LINK", Value: generateRepo.Link(), Transformers: nil}, | 		{Name: "REPO_LINK", Value: generateRepo.Link(), Transformers: nil}, | ||||||
| 		{Name: "TEMPLATE_LINK", Value: templateRepo.Link(), Transformers: nil}, | 		{Name: "TEMPLATE_LINK", Value: templateRepo.Link(), Transformers: nil}, | ||||||
| 		{Name: "REPO_HTTPS_URL", Value: generateRepo.CloneLink().HTTPS, Transformers: nil}, | 		{Name: "REPO_HTTPS_URL", Value: generateRepo.CloneLinkGeneral(ctx).HTTPS, Transformers: nil}, | ||||||
| 		{Name: "TEMPLATE_HTTPS_URL", Value: templateRepo.CloneLink().HTTPS, Transformers: nil}, | 		{Name: "TEMPLATE_HTTPS_URL", Value: templateRepo.CloneLinkGeneral(ctx).HTTPS, Transformers: nil}, | ||||||
| 		{Name: "REPO_SSH_URL", Value: generateRepo.CloneLink().SSH, Transformers: nil}, | 		{Name: "REPO_SSH_URL", Value: generateRepo.CloneLinkGeneral(ctx).SSH, Transformers: nil}, | ||||||
| 		{Name: "TEMPLATE_SSH_URL", Value: templateRepo.CloneLink().SSH, Transformers: nil}, | 		{Name: "TEMPLATE_SSH_URL", Value: templateRepo.CloneLinkGeneral(ctx).SSH, Transformers: nil}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	expansionMap := make(map[string]string) | 	expansionMap := make(map[string]string) | ||||||
| @@ -138,7 +138,7 @@ func readGiteaTemplateFile(tmpDir string) (*GiteaTemplate, error) { | |||||||
| 	return &GiteaTemplate{Path: gtPath, Content: content}, nil | 	return &GiteaTemplate{Path: gtPath, Content: content}, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func processGiteaTemplateFile(tmpDir string, templateRepo, generateRepo *repo_model.Repository, giteaTemplateFile *GiteaTemplate) error { | func processGiteaTemplateFile(ctx context.Context, tmpDir string, templateRepo, generateRepo *repo_model.Repository, giteaTemplateFile *GiteaTemplate) error { | ||||||
| 	if err := util.Remove(giteaTemplateFile.Path); err != nil { | 	if err := util.Remove(giteaTemplateFile.Path); err != nil { | ||||||
| 		return fmt.Errorf("remove .giteatemplate: %w", err) | 		return fmt.Errorf("remove .giteatemplate: %w", err) | ||||||
| 	} | 	} | ||||||
| @@ -163,12 +163,12 @@ func processGiteaTemplateFile(tmpDir string, templateRepo, generateRepo *repo_mo | |||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				generatedContent := []byte(generateExpansion(string(content), templateRepo, generateRepo, false)) | 				generatedContent := []byte(generateExpansion(ctx, string(content), templateRepo, generateRepo, false)) | ||||||
| 				if err := os.WriteFile(path, generatedContent, 0o644); err != nil { | 				if err := os.WriteFile(path, generatedContent, 0o644); err != nil { | ||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				substPath := filepath.FromSlash(filepath.Join(tmpDirSlash, generateExpansion(base, templateRepo, generateRepo, true))) | 				substPath := filepath.FromSlash(filepath.Join(tmpDirSlash, generateExpansion(ctx, base, templateRepo, generateRepo, true))) | ||||||
|  |  | ||||||
| 				// Create parent subdirectories if needed or continue silently if it exists | 				// Create parent subdirectories if needed or continue silently if it exists | ||||||
| 				if err = os.MkdirAll(filepath.Dir(substPath), 0o755); err != nil { | 				if err = os.MkdirAll(filepath.Dir(substPath), 0o755); err != nil { | ||||||
| @@ -226,7 +226,7 @@ func generateRepoCommit(ctx context.Context, repo, templateRepo, generateRepo *r | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if giteaTemplateFile != nil { | 	if giteaTemplateFile != nil { | ||||||
| 		err = processGiteaTemplateFile(tmpDir, templateRepo, generateRepo, giteaTemplateFile) | 		err = processGiteaTemplateFile(ctx, tmpDir, templateRepo, generateRepo, giteaTemplateFile) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -66,7 +66,7 @@ func TestDumpRestore(t *testing.T) { | |||||||
| 			Milestones:     true, | 			Milestones:     true, | ||||||
| 			Comments:       true, | 			Comments:       true, | ||||||
| 			AuthToken:      token, | 			AuthToken:      token, | ||||||
| 			CloneAddr:      repo.CloneLink().HTTPS, | 			CloneAddr:      repo.CloneLinkGeneral(context.Background()).HTTPS, | ||||||
| 			RepoName:       reponame, | 			RepoName:       reponame, | ||||||
| 		} | 		} | ||||||
| 		err = migrations.DumpRepository(ctx, basePath, repoOwner.Name, opts) | 		err = migrations.DumpRepository(ctx, basePath, repoOwner.Name, opts) | ||||||
| @@ -96,7 +96,7 @@ func TestDumpRestore(t *testing.T) { | |||||||
| 		// Phase 3: dump restored from the Gitea instance to the filesystem | 		// Phase 3: dump restored from the Gitea instance to the filesystem | ||||||
| 		// | 		// | ||||||
| 		opts.RepoName = newreponame | 		opts.RepoName = newreponame | ||||||
| 		opts.CloneAddr = newrepo.CloneLink().HTTPS | 		opts.CloneAddr = newrepo.CloneLinkGeneral(context.Background()).HTTPS | ||||||
| 		err = migrations.DumpRepository(ctx, basePath, repoOwner.Name, opts) | 		err = migrations.DumpRepository(ctx, basePath, repoOwner.Name, opts) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user