You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

blob_nogogit.go 2.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. // Copyright 2020 The Gitea Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. //go:build !gogit
  5. package git
  6. import (
  7. "bufio"
  8. "bytes"
  9. "io"
  10. "math"
  11. "code.gitea.io/gitea/modules/log"
  12. )
  13. // Blob represents a Git object.
  14. type Blob struct {
  15. ID SHA1
  16. gotSize bool
  17. size int64
  18. name string
  19. repo *Repository
  20. }
  21. // DataAsync gets a ReadCloser for the contents of a blob without reading it all.
  22. // Calling the Close function on the result will discard all unread output.
  23. func (b *Blob) DataAsync() (io.ReadCloser, error) {
  24. wr, rd, cancel := b.repo.CatFileBatch(b.repo.Ctx)
  25. _, err := wr.Write([]byte(b.ID.String() + "\n"))
  26. if err != nil {
  27. cancel()
  28. return nil, err
  29. }
  30. _, _, size, err := ReadBatchLine(rd)
  31. if err != nil {
  32. cancel()
  33. return nil, err
  34. }
  35. b.gotSize = true
  36. b.size = size
  37. if size < 4096 {
  38. bs, err := io.ReadAll(io.LimitReader(rd, size))
  39. defer cancel()
  40. if err != nil {
  41. return nil, err
  42. }
  43. _, err = rd.Discard(1)
  44. return io.NopCloser(bytes.NewReader(bs)), err
  45. }
  46. return &blobReader{
  47. rd: rd,
  48. n: size,
  49. cancel: cancel,
  50. }, nil
  51. }
  52. // Size returns the uncompressed size of the blob
  53. func (b *Blob) Size() int64 {
  54. if b.gotSize {
  55. return b.size
  56. }
  57. wr, rd, cancel := b.repo.CatFileBatchCheck(b.repo.Ctx)
  58. defer cancel()
  59. _, err := wr.Write([]byte(b.ID.String() + "\n"))
  60. if err != nil {
  61. log.Debug("error whilst reading size for %s in %s. Error: %v", b.ID.String(), b.repo.Path, err)
  62. return 0
  63. }
  64. _, _, b.size, err = ReadBatchLine(rd)
  65. if err != nil {
  66. log.Debug("error whilst reading size for %s in %s. Error: %v", b.ID.String(), b.repo.Path, err)
  67. return 0
  68. }
  69. b.gotSize = true
  70. return b.size
  71. }
  72. type blobReader struct {
  73. rd *bufio.Reader
  74. n int64
  75. cancel func()
  76. }
  77. func (b *blobReader) Read(p []byte) (n int, err error) {
  78. if b.n <= 0 {
  79. return 0, io.EOF
  80. }
  81. if int64(len(p)) > b.n {
  82. p = p[0:b.n]
  83. }
  84. n, err = b.rd.Read(p)
  85. b.n -= int64(n)
  86. return
  87. }
  88. // Close implements io.Closer
  89. func (b *blobReader) Close() error {
  90. defer b.cancel()
  91. if b.n > 0 {
  92. for b.n > math.MaxInt32 {
  93. n, err := b.rd.Discard(math.MaxInt32)
  94. b.n -= int64(n)
  95. if err != nil {
  96. return err
  97. }
  98. b.n -= math.MaxInt32
  99. }
  100. n, err := b.rd.Discard(int(b.n))
  101. b.n -= int64(n)
  102. if err != nil {
  103. return err
  104. }
  105. }
  106. if b.n == 0 {
  107. _, err := b.rd.Discard(1)
  108. b.n--
  109. return err
  110. }
  111. return nil
  112. }