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.3KB

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