mirror of
https://github.com/go-gitea/gitea.git
synced 2025-11-10 02:57:14 -05:00
Fix #35852, the root problem is that the "name" field is heavily abused (since #6816, and no way to get a clear fix) There are still a lot of legacy problems in old code. Co-authored-by: Giteabot <teabot@gitea.io>
173 lines
4.3 KiB
Go
173 lines
4.3 KiB
Go
// Copyright 2015 The Gogs Authors. All rights reserved.
|
|
// Copyright 2019 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package git
|
|
|
|
import (
|
|
"path"
|
|
"slices"
|
|
"strings"
|
|
|
|
"code.gitea.io/gitea/modules/util"
|
|
)
|
|
|
|
// TreeEntry the leaf in the git tree
|
|
type TreeEntry struct {
|
|
ID ObjectID
|
|
|
|
name string
|
|
ptree *Tree
|
|
|
|
entryMode EntryMode
|
|
|
|
size int64
|
|
sized bool
|
|
}
|
|
|
|
// Name returns the name of the entry (base name)
|
|
func (te *TreeEntry) Name() string {
|
|
return te.name
|
|
}
|
|
|
|
// Mode returns the mode of the entry
|
|
func (te *TreeEntry) Mode() EntryMode {
|
|
return te.entryMode
|
|
}
|
|
|
|
// IsSubModule if the entry is a submodule
|
|
func (te *TreeEntry) IsSubModule() bool {
|
|
return te.entryMode.IsSubModule()
|
|
}
|
|
|
|
// IsDir if the entry is a sub dir
|
|
func (te *TreeEntry) IsDir() bool {
|
|
return te.entryMode.IsDir()
|
|
}
|
|
|
|
// IsLink if the entry is a symlink
|
|
func (te *TreeEntry) IsLink() bool {
|
|
return te.entryMode.IsLink()
|
|
}
|
|
|
|
// IsRegular if the entry is a regular file
|
|
func (te *TreeEntry) IsRegular() bool {
|
|
return te.entryMode.IsRegular()
|
|
}
|
|
|
|
// IsExecutable if the entry is an executable file (not necessarily binary)
|
|
func (te *TreeEntry) IsExecutable() bool {
|
|
return te.entryMode.IsExecutable()
|
|
}
|
|
|
|
// Type returns the type of the entry (commit, tree, blob)
|
|
func (te *TreeEntry) Type() string {
|
|
switch te.Mode() {
|
|
case EntryModeCommit:
|
|
return "commit"
|
|
case EntryModeTree:
|
|
return "tree"
|
|
default:
|
|
return "blob"
|
|
}
|
|
}
|
|
|
|
type EntryFollowResult struct {
|
|
SymlinkContent string
|
|
TargetFullPath string
|
|
TargetEntry *TreeEntry
|
|
}
|
|
|
|
func EntryFollowLink(commit *Commit, fullPath string, te *TreeEntry) (*EntryFollowResult, error) {
|
|
if !te.IsLink() {
|
|
return nil, util.ErrorWrap(util.ErrUnprocessableContent, "%q is not a symlink", fullPath)
|
|
}
|
|
|
|
// git's filename max length is 4096, hopefully a link won't be longer than multiple of that
|
|
const maxSymlinkSize = 20 * 4096
|
|
if te.Blob().Size() > maxSymlinkSize {
|
|
return nil, util.ErrorWrap(util.ErrUnprocessableContent, "%q content exceeds symlink limit", fullPath)
|
|
}
|
|
|
|
link, err := te.Blob().GetBlobContent(maxSymlinkSize)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if strings.HasPrefix(link, "/") {
|
|
// It's said that absolute path will be stored as is in Git
|
|
return &EntryFollowResult{SymlinkContent: link}, util.ErrorWrap(util.ErrUnprocessableContent, "%q is an absolute symlink", fullPath)
|
|
}
|
|
|
|
targetFullPath := path.Join(path.Dir(fullPath), link)
|
|
targetEntry, err := commit.GetTreeEntryByPath(targetFullPath)
|
|
if err != nil {
|
|
return &EntryFollowResult{SymlinkContent: link}, err
|
|
}
|
|
return &EntryFollowResult{SymlinkContent: link, TargetFullPath: targetFullPath, TargetEntry: targetEntry}, nil
|
|
}
|
|
|
|
func EntryFollowLinks(commit *Commit, firstFullPath string, firstTreeEntry *TreeEntry, optLimit ...int) (res *EntryFollowResult, err error) {
|
|
limit := util.OptionalArg(optLimit, 10)
|
|
treeEntry, fullPath := firstTreeEntry, firstFullPath
|
|
for range limit {
|
|
res, err = EntryFollowLink(commit, fullPath, treeEntry)
|
|
if err != nil {
|
|
return res, err
|
|
}
|
|
treeEntry, fullPath = res.TargetEntry, res.TargetFullPath
|
|
if !treeEntry.IsLink() {
|
|
break
|
|
}
|
|
}
|
|
if treeEntry.IsLink() {
|
|
return res, util.ErrorWrap(util.ErrUnprocessableContent, "%q has too many links", firstFullPath)
|
|
}
|
|
return res, nil
|
|
}
|
|
|
|
// returns the Tree pointed to by this TreeEntry, or nil if this is not a tree
|
|
func (te *TreeEntry) Tree() *Tree {
|
|
t, err := te.ptree.repo.getTree(te.ID)
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
t.ptree = te.ptree
|
|
return t
|
|
}
|
|
|
|
// GetSubJumpablePathName return the full path of subdirectory jumpable ( contains only one directory )
|
|
func (te *TreeEntry) GetSubJumpablePathName() string {
|
|
if te.IsSubModule() || !te.IsDir() {
|
|
return ""
|
|
}
|
|
tree, err := te.ptree.SubTree(te.Name())
|
|
if err != nil {
|
|
return te.Name()
|
|
}
|
|
entries, _ := tree.ListEntries()
|
|
if len(entries) == 1 && entries[0].IsDir() {
|
|
name := entries[0].GetSubJumpablePathName()
|
|
if name != "" {
|
|
return te.Name() + "/" + name
|
|
}
|
|
}
|
|
return te.Name()
|
|
}
|
|
|
|
// Entries a list of entry
|
|
type Entries []*TreeEntry
|
|
|
|
// CustomSort customizable string comparing sort entry list
|
|
func (tes Entries) CustomSort(cmp func(s1, s2 string) int) {
|
|
slices.SortFunc(tes, func(a, b *TreeEntry) int {
|
|
s1Dir, s2Dir := a.IsDir() || a.IsSubModule(), b.IsDir() || b.IsSubModule()
|
|
if s1Dir != s2Dir {
|
|
if s1Dir {
|
|
return -1
|
|
}
|
|
return 1
|
|
}
|
|
return cmp(a.Name(), b.Name())
|
|
})
|
|
}
|