gitea源码

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