mirror of
https://github.com/go-gitea/gitea.git
synced 2025-10-24 03:30:15 +02:00
184 lines
4.7 KiB
Go
184 lines
4.7 KiB
Go
// Copyright 2022 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package url
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net"
|
|
stdurl "net/url"
|
|
"strings"
|
|
|
|
"code.gitea.io/gitea/modules/httplib"
|
|
"code.gitea.io/gitea/modules/setting"
|
|
"code.gitea.io/gitea/modules/util"
|
|
)
|
|
|
|
// ErrWrongURLFormat represents an error with wrong url format
|
|
type ErrWrongURLFormat struct {
|
|
URL string
|
|
}
|
|
|
|
func (err ErrWrongURLFormat) Error() string {
|
|
return fmt.Sprintf("git URL %s format is wrong", err.URL)
|
|
}
|
|
|
|
// GitURL represents a git URL
|
|
type GitURL struct {
|
|
*stdurl.URL
|
|
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
|
|
func (u *GitURL) String() string {
|
|
switch u.extraMark {
|
|
case 0:
|
|
return u.URL.String()
|
|
case 1:
|
|
return fmt.Sprintf("%s@%s:%s", u.User.Username(), u.Host, u.Path)
|
|
case 2:
|
|
return u.Path
|
|
default:
|
|
return ""
|
|
}
|
|
}
|
|
|
|
// ParseGitURL parse all kinds of git URL:
|
|
// * 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, "://") {
|
|
u, err := stdurl.Parse(remote)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &GitURL{URL: u}, nil
|
|
} else if strings.Contains(remote, "@") && strings.Contains(remote, ":") {
|
|
url := stdurl.URL{
|
|
Scheme: "ssh",
|
|
}
|
|
squareBrackets := false
|
|
lastIndex := -1
|
|
FOR:
|
|
for i := 0; i < len(remote); i++ {
|
|
switch remote[i] {
|
|
case '@':
|
|
url.User = stdurl.User(remote[:i])
|
|
lastIndex = i + 1
|
|
case ':':
|
|
if !squareBrackets {
|
|
url.Host = strings.ReplaceAll(remote[lastIndex:i], "%25", "%")
|
|
if len(remote) <= i+1 {
|
|
return nil, ErrWrongURLFormat{URL: remote}
|
|
}
|
|
url.Path = remote[i+1:]
|
|
break FOR
|
|
}
|
|
case '[':
|
|
squareBrackets = true
|
|
case ']':
|
|
squareBrackets = false
|
|
}
|
|
}
|
|
return &GitURL{
|
|
URL: &url,
|
|
extraMark: 1,
|
|
}, nil
|
|
}
|
|
|
|
return &GitURL{
|
|
URL: &stdurl.URL{
|
|
Scheme: "file",
|
|
Path: remote,
|
|
},
|
|
extraMark: 2,
|
|
}, nil
|
|
}
|
|
|
|
type RepositoryURL struct {
|
|
GitURL *GitURL
|
|
|
|
// if the URL belongs to current Gitea instance, then the below fields have values
|
|
OwnerName string
|
|
RepoName string
|
|
RemainingPath string
|
|
}
|
|
|
|
// ParseRepositoryURL tries to parse a Git URL and extract the owner/repository name if it belongs to current Gitea instance.
|
|
func ParseRepositoryURL(ctx context.Context, repoURL string) (*RepositoryURL, error) {
|
|
// 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]
|
|
parsed, err := ParseGitURL(repoURL)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ret := &RepositoryURL{}
|
|
ret.GitURL = parsed
|
|
|
|
fillPathParts := func(s string) {
|
|
s = strings.TrimPrefix(s, "/")
|
|
fields := strings.SplitN(s, "/", 3)
|
|
if len(fields) >= 2 {
|
|
ret.OwnerName = fields[0]
|
|
ret.RepoName = strings.TrimSuffix(fields[1], ".git")
|
|
if len(fields) == 3 {
|
|
ret.RemainingPath = "/" + fields[2]
|
|
}
|
|
}
|
|
}
|
|
|
|
switch parsed.URL.Scheme {
|
|
case "http", "https":
|
|
if !httplib.IsCurrentGiteaSiteURL(ctx, repoURL) {
|
|
return ret, nil
|
|
}
|
|
fillPathParts(strings.TrimPrefix(parsed.URL.Path, setting.AppSubURL))
|
|
case "ssh", "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, nil
|
|
}
|
|
// 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, nil
|
|
}
|
|
|
|
// MakeRepositoryWebLink generates a web link (http/https) for a git repository (by guessing sometimes)
|
|
func MakeRepositoryWebLink(repoURL *RepositoryURL) string {
|
|
if repoURL.OwnerName != "" {
|
|
return setting.AppSubURL + "/" + repoURL.OwnerName + "/" + repoURL.RepoName
|
|
}
|
|
|
|
// now, let's guess, for example:
|
|
// * git@github.com:owner/submodule.git
|
|
// * https://github.com/example/submodule1.git
|
|
switch repoURL.GitURL.Scheme {
|
|
case "http", "https":
|
|
return strings.TrimSuffix(repoURL.GitURL.String(), ".git")
|
|
case "ssh", "git+ssh":
|
|
hostname, _, _ := net.SplitHostPort(repoURL.GitURL.Host)
|
|
hostname = util.IfZero(hostname, repoURL.GitURL.Host)
|
|
urlPath := strings.TrimSuffix(repoURL.GitURL.Path, ".git")
|
|
urlPath = strings.TrimPrefix(urlPath, "/")
|
|
urlFull := fmt.Sprintf("https://%s/%s", hostname, urlPath)
|
|
urlFull = strings.TrimSuffix(urlFull, "/")
|
|
return urlFull
|
|
}
|
|
return ""
|
|
}
|