2020-12-17 14:00:47 +00:00
// Copyright 2020 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
// +build !gogit
package git
import (
"bufio"
2021-05-10 02:27:03 +01:00
"bytes"
2020-12-17 14:00:47 +00:00
"io"
2021-05-10 02:27:03 +01:00
"io/ioutil"
"math"
2021-06-25 18:54:08 +02:00
"code.gitea.io/gitea/modules/log"
2020-12-17 14:00:47 +00:00
)
// Blob represents a Git object.
type Blob struct {
ID SHA1
2021-05-10 02:27:03 +01:00
gotSize bool
size int64
name string
repo * Repository
2020-12-17 14:00:47 +00:00
}
// DataAsync gets a ReadCloser for the contents of a blob without reading it all.
// Calling the Close function on the result will discard all unread output.
func ( b * Blob ) DataAsync ( ) ( io . ReadCloser , error ) {
2021-05-10 02:27:03 +01:00
wr , rd , cancel := b . repo . CatFileBatch ( )
2020-12-17 14:00:47 +00:00
2021-05-10 02:27:03 +01:00
_ , err := wr . Write ( [ ] byte ( b . ID . String ( ) + "\n" ) )
2020-12-17 14:00:47 +00:00
if err != nil {
2021-05-10 02:27:03 +01:00
cancel ( )
2020-12-17 14:00:47 +00:00
return nil , err
}
2021-05-10 02:27:03 +01:00
_ , _ , size , err := ReadBatchLine ( rd )
if err != nil {
cancel ( )
return nil , err
}
b . gotSize = true
b . size = size
2020-12-17 14:00:47 +00:00
2021-05-10 02:27:03 +01:00
if size < 4096 {
bs , err := ioutil . ReadAll ( io . LimitReader ( rd , size ) )
if err != nil {
cancel ( )
return nil , err
}
_ , err = rd . Discard ( 1 )
return ioutil . NopCloser ( bytes . NewReader ( bs ) ) , err
}
return & blobReader {
rd : rd ,
n : size ,
cancel : cancel ,
2021-02-03 22:36:38 +01:00
} , nil
2020-12-17 14:00:47 +00:00
}
// Size returns the uncompressed size of the blob
func ( b * Blob ) Size ( ) int64 {
if b . gotSize {
return b . size
}
2021-05-10 02:27:03 +01:00
wr , rd , cancel := b . repo . CatFileBatchCheck ( )
defer cancel ( )
_ , err := wr . Write ( [ ] byte ( b . ID . String ( ) + "\n" ) )
2020-12-17 14:00:47 +00:00
if err != nil {
2021-06-25 18:54:08 +02:00
log . Debug ( "error whilst reading size for %s in %s. Error: %v" , b . ID . String ( ) , b . repo . Path , err )
2020-12-17 14:00:47 +00:00
return 0
}
2021-05-10 02:27:03 +01:00
_ , _ , b . size , err = ReadBatchLine ( rd )
2020-12-17 14:00:47 +00:00
if err != nil {
2021-06-25 18:54:08 +02:00
log . Debug ( "error whilst reading size for %s in %s. Error: %v" , b . ID . String ( ) , b . repo . Path , err )
2020-12-17 14:00:47 +00:00
return 0
}
2021-05-10 02:27:03 +01:00
2020-12-17 14:00:47 +00:00
b . gotSize = true
return b . size
}
2021-05-10 02:27:03 +01:00
type blobReader struct {
rd * bufio . Reader
n int64
cancel func ( )
}
func ( b * blobReader ) Read ( p [ ] byte ) ( n int , err error ) {
if b . n <= 0 {
return 0 , io . EOF
}
if int64 ( len ( p ) ) > b . n {
p = p [ 0 : b . n ]
}
n , err = b . rd . Read ( p )
b . n -= int64 ( n )
return
}
// Close implements io.Closer
func ( b * blobReader ) Close ( ) error {
if b . n > 0 {
for b . n > math . MaxInt32 {
n , err := b . rd . Discard ( math . MaxInt32 )
b . n -= int64 ( n )
if err != nil {
b . cancel ( )
return err
}
b . n -= math . MaxInt32
}
n , err := b . rd . Discard ( int ( b . n ) )
b . n -= int64 ( n )
if err != nil {
b . cancel ( )
return err
}
}
if b . n == 0 {
_ , err := b . rd . Discard ( 1 )
b . n --
b . cancel ( )
return err
}
return nil
}