2020-12-17 17:00:47 +03:00
// Copyright 2020 The Gitea Authors. All rights reserved.
2022-11-27 21:20:29 +03:00
// SPDX-License-Identifier: MIT
2020-12-17 17:00:47 +03:00
2021-08-24 19:47:09 +03:00
//go:build !gogit
2020-12-17 17:00:47 +03:00
package git
import (
2021-05-10 04:27:03 +03:00
"io"
"math"
2020-12-17 17:00:47 +03:00
"strings"
)
// Tree represents a flat directory listing.
type Tree struct {
ID SHA1
ResolvedID SHA1
repo * Repository
// parent tree
ptree * Tree
entries Entries
entriesParsed bool
entriesRecursive Entries
entriesRecursiveParsed bool
}
// ListEntries returns all entries of current tree.
func ( t * Tree ) ListEntries ( ) ( Entries , error ) {
if t . entriesParsed {
return t . entries , nil
}
2021-05-10 04:27:03 +03:00
if t . repo != nil {
2021-11-30 23:06:32 +03:00
wr , rd , cancel := t . repo . CatFileBatch ( t . repo . Ctx )
2021-05-10 04:27:03 +03:00
defer cancel ( )
_ , _ = wr . Write ( [ ] byte ( t . ID . String ( ) + "\n" ) )
_ , typ , sz , err := ReadBatchLine ( rd )
if err != nil {
return nil , err
}
if typ == "commit" {
treeID , err := ReadTreeID ( rd , sz )
if err != nil && err != io . EOF {
return nil , err
}
_ , _ = wr . Write ( [ ] byte ( treeID + "\n" ) )
_ , typ , sz , err = ReadBatchLine ( rd )
if err != nil {
return nil , err
}
}
if typ == "tree" {
t . entries , err = catBatchParseTreeEntries ( t , rd , sz )
if err != nil {
return nil , err
}
t . entriesParsed = true
return t . entries , nil
}
// Not a tree just use ls-tree instead
for sz > math . MaxInt32 {
discarded , err := rd . Discard ( math . MaxInt32 )
sz -= int64 ( discarded )
if err != nil {
return nil , err
}
}
for sz > 0 {
discarded , err := rd . Discard ( int ( sz ) )
sz -= int64 ( discarded )
if err != nil {
return nil , err
}
}
}
2022-10-23 17:44:45 +03:00
stdout , _ , runErr := NewCommand ( t . repo . Ctx , "ls-tree" , "-l" ) . AddDynamicArguments ( t . ID . String ( ) ) . RunStdBytes ( & RunOpts { Dir : t . repo . Path } )
2022-04-01 05:55:30 +03:00
if runErr != nil {
if strings . Contains ( runErr . Error ( ) , "fatal: Not a valid object name" ) || strings . Contains ( runErr . Error ( ) , "fatal: not a tree object" ) {
2020-12-17 17:00:47 +03:00
return nil , ErrNotExist {
ID : t . ID . String ( ) ,
}
}
2022-04-01 05:55:30 +03:00
return nil , runErr
2020-12-17 17:00:47 +03:00
}
2022-04-01 05:55:30 +03:00
var err error
2020-12-17 17:00:47 +03:00
t . entries , err = parseTreeEntries ( stdout , t )
if err == nil {
t . entriesParsed = true
}
return t . entries , err
}
2022-10-07 20:20:53 +03:00
// listEntriesRecursive returns all entries of current tree recursively including all subtrees
// extraArgs could be "-l" to get the size, which is slower
2022-10-23 17:44:45 +03:00
func ( t * Tree ) listEntriesRecursive ( extraArgs ... CmdArg ) ( Entries , error ) {
2020-12-17 17:00:47 +03:00
if t . entriesRecursiveParsed {
return t . entriesRecursive , nil
}
2022-04-01 05:55:30 +03:00
2022-10-23 17:44:45 +03:00
args := append ( [ ] CmdArg { "ls-tree" , "-t" , "-r" } , extraArgs ... )
args = append ( args , CmdArg ( t . ID . String ( ) ) )
2022-10-07 20:20:53 +03:00
stdout , _ , runErr := NewCommand ( t . repo . Ctx , args ... ) . RunStdBytes ( & RunOpts { Dir : t . repo . Path } )
2022-04-01 05:55:30 +03:00
if runErr != nil {
return nil , runErr
2020-12-17 17:00:47 +03:00
}
2022-04-01 05:55:30 +03:00
var err error
2020-12-17 17:00:47 +03:00
t . entriesRecursive , err = parseTreeEntries ( stdout , t )
if err == nil {
t . entriesRecursiveParsed = true
}
return t . entriesRecursive , err
}
2022-10-07 20:20:53 +03:00
// ListEntriesRecursiveFast returns all entries of current tree recursively including all subtrees, no size
func ( t * Tree ) ListEntriesRecursiveFast ( ) ( Entries , error ) {
return t . listEntriesRecursive ( )
}
// ListEntriesRecursiveWithSize returns all entries of current tree recursively including all subtrees, with size
func ( t * Tree ) ListEntriesRecursiveWithSize ( ) ( Entries , error ) {
return t . listEntriesRecursive ( "--long" )
}