mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-31 11:35:03 +01: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 ""
 | |
| }
 |