2018-11-27 23:52:20 +02:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2022-11-27 13:20:29 -05:00
// SPDX-License-Identifier: MIT
2018-11-27 23:52:20 +02:00
package git
2022-09-04 11:47:56 +01:00
import (
"regexp"
"strings"
2023-05-26 09:04:48 +08:00
"code.gitea.io/gitea/modules/util"
2022-09-04 11:47:56 +01:00
)
2020-11-08 17:21:54 +00:00
2021-12-02 08:28:08 +01:00
const (
// RemotePrefix is the base directory of the remotes information of git.
RemotePrefix = "refs/remotes/"
// PullPrefix is the base directory of the pull information of git.
PullPrefix = "refs/pull/"
)
2022-09-04 11:47:56 +01:00
// refNamePatternInvalid is regular expression with unallowed characters in git reference name
// They cannot have ASCII control characters (i.e. bytes whose values are lower than \040, or \177 DEL), space, tilde ~, caret ^, or colon : anywhere.
// They cannot have question-mark ?, asterisk *, or open bracket [ anywhere
var refNamePatternInvalid = regexp . MustCompile (
` [\000-\037\177 \\~^:?*[]| ` + // No absolutely invalid characters
` (?:^[/.])| ` + // Not HasPrefix("/") or "."
` (?:/\.)| ` + // no "/."
` (?:\.lock$)|(?:\.lock/)| ` + // No ".lock/"" or ".lock" at the end
` (?:\.\.)| ` + // no ".." anywhere
` (?://)| ` + // no "//" anywhere
` (?:@ { )| ` + // no "@{"
` (?:[/.]$)| ` + // no terminal '/' or '.'
` (?:^@$) ` ) // Not "@"
// IsValidRefPattern ensures that the provided string could be a valid reference
func IsValidRefPattern ( name string ) bool {
return ! refNamePatternInvalid . MatchString ( name )
}
func SanitizeRefPattern ( name string ) string {
return refNamePatternInvalid . ReplaceAllString ( name , "_" )
}
2018-11-27 23:52:20 +02:00
// Reference represents a Git ref.
type Reference struct {
Name string
repo * Repository
2023-12-13 21:02:00 +00:00
Object ObjectID // The id of this commit object
2018-11-27 23:52:20 +02:00
Type string
}
// Commit return the commit of the reference
func ( ref * Reference ) Commit ( ) ( * Commit , error ) {
return ref . repo . getCommit ( ref . Object )
}
2020-11-08 17:21:54 +00:00
2023-05-30 21:36:58 +08:00
// ShortName returns the short name of the reference
func ( ref * Reference ) ShortName ( ) string {
return RefName ( ref . Name ) . ShortName ( )
}
2022-12-01 19:56:04 +08:00
// RefGroup returns the group type of the reference
func ( ref * Reference ) RefGroup ( ) string {
return RefName ( ref . Name ) . RefGroup ( )
}
2023-05-26 09:04:48 +08:00
// ForPrefix special ref to create a pull request: refs/for/<target-branch>/<topic-branch>
// or refs/for/<targe-branch> -o topic='<topic-branch>'
const ForPrefix = "refs/for/"
// TODO: /refs/for-review for suggest change interface
// RefName represents a full git reference name
2022-12-01 19:56:04 +08:00
type RefName string
2023-05-26 09:04:48 +08:00
func RefNameFromBranch ( shortName string ) RefName {
return RefName ( BranchPrefix + shortName )
}
func RefNameFromTag ( shortName string ) RefName {
return RefName ( TagPrefix + shortName )
}
func ( ref RefName ) String ( ) string {
return string ( ref )
}
2022-12-01 19:56:04 +08:00
func ( ref RefName ) IsBranch ( ) bool {
return strings . HasPrefix ( string ( ref ) , BranchPrefix )
}
func ( ref RefName ) IsTag ( ) bool {
return strings . HasPrefix ( string ( ref ) , TagPrefix )
}
2023-05-26 09:04:48 +08:00
func ( ref RefName ) IsRemote ( ) bool {
return strings . HasPrefix ( string ( ref ) , RemotePrefix )
}
func ( ref RefName ) IsPull ( ) bool {
return strings . HasPrefix ( string ( ref ) , PullPrefix ) && strings . IndexByte ( string ( ref ) [ len ( PullPrefix ) : ] , '/' ) > - 1
}
func ( ref RefName ) IsFor ( ) bool {
return strings . HasPrefix ( string ( ref ) , ForPrefix )
}
func ( ref RefName ) nameWithoutPrefix ( prefix string ) string {
if strings . HasPrefix ( string ( ref ) , prefix ) {
return strings . TrimPrefix ( string ( ref ) , prefix )
}
return ""
}
// TagName returns simple tag name if it's an operation to a tag
func ( ref RefName ) TagName ( ) string {
return ref . nameWithoutPrefix ( TagPrefix )
}
// BranchName returns simple branch name if it's an operation to branch
func ( ref RefName ) BranchName ( ) string {
return ref . nameWithoutPrefix ( BranchPrefix )
}
// PullName returns the pull request name part of refs like refs/pull/<pull_name>/head
func ( ref RefName ) PullName ( ) string {
refName := string ( ref )
lastIdx := strings . LastIndexByte ( refName [ len ( PullPrefix ) : ] , '/' )
if strings . HasPrefix ( refName , PullPrefix ) && lastIdx > - 1 {
return refName [ len ( PullPrefix ) : lastIdx + len ( PullPrefix ) ]
}
return ""
}
// ForBranchName returns the branch name part of refs like refs/for/<branch_name>
func ( ref RefName ) ForBranchName ( ) string {
return ref . nameWithoutPrefix ( ForPrefix )
}
func ( ref RefName ) RemoteName ( ) string {
return ref . nameWithoutPrefix ( RemotePrefix )
}
2022-12-01 19:56:04 +08:00
// ShortName returns the short name of the reference name
func ( ref RefName ) ShortName ( ) string {
refName := string ( ref )
2023-05-26 09:04:48 +08:00
if ref . IsBranch ( ) {
return ref . BranchName ( )
}
if ref . IsTag ( ) {
return ref . TagName ( )
2020-11-08 17:21:54 +00:00
}
2023-05-26 09:04:48 +08:00
if ref . IsRemote ( ) {
return ref . RemoteName ( )
2020-11-08 17:21:54 +00:00
}
2023-05-26 09:04:48 +08:00
if ref . IsPull ( ) {
return ref . PullName ( )
2020-11-08 17:21:54 +00:00
}
2023-05-26 09:04:48 +08:00
if ref . IsFor ( ) {
return ref . ForBranchName ( )
2020-11-08 17:21:54 +00:00
}
2022-12-01 19:56:04 +08:00
return refName
2020-11-08 17:21:54 +00:00
}
// RefGroup returns the group type of the reference
2023-06-13 14:05:28 +08:00
// Using the name of the directory under .git/refs
2022-12-01 19:56:04 +08:00
func ( ref RefName ) RefGroup ( ) string {
2023-05-26 09:04:48 +08:00
if ref . IsBranch ( ) {
2020-11-08 17:21:54 +00:00
return "heads"
}
2023-05-26 09:04:48 +08:00
if ref . IsTag ( ) {
2020-11-08 17:21:54 +00:00
return "tags"
}
2023-05-26 09:04:48 +08:00
if ref . IsRemote ( ) {
2020-11-08 17:21:54 +00:00
return "remotes"
}
2023-05-26 09:04:48 +08:00
if ref . IsPull ( ) {
2020-11-08 17:21:54 +00:00
return "pull"
}
2023-05-26 09:04:48 +08:00
if ref . IsFor ( ) {
return "for"
}
2020-11-08 17:21:54 +00:00
return ""
}
2023-05-26 09:04:48 +08:00
2023-06-13 14:05:28 +08:00
// RefType returns the simple ref type of the reference, e.g. branch, tag
2024-04-27 10:03:49 +02:00
// It's different from RefGroup, which is using the name of the directory under .git/refs
2023-06-13 14:05:28 +08:00
// Here we using branch but not heads, using tag but not tags
func ( ref RefName ) RefType ( ) string {
var refType string
if ref . IsBranch ( ) {
refType = "branch"
} else if ref . IsTag ( ) {
refType = "tag"
}
return refType
}
2023-05-26 09:04:48 +08:00
// RefURL returns the absolute URL for a ref in a repository
func RefURL ( repoURL , ref string ) string {
refFullName := RefName ( ref )
refName := util . PathEscapeSegments ( refFullName . ShortName ( ) )
switch {
case refFullName . IsBranch ( ) :
return repoURL + "/src/branch/" + refName
case refFullName . IsTag ( ) :
return repoURL + "/src/tag/" + refName
2023-12-17 19:56:08 +08:00
case ! Sha1ObjectFormat . IsValid ( ref ) :
2023-05-26 09:04:48 +08:00
// assume they mean a branch
return repoURL + "/src/branch/" + refName
default :
return repoURL + "/src/commit/" + refName
}
}