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

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