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.

download.go 3.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. // Copyright 2014 The Gogs Authors. All rights reserved.
  2. // Copyright 2018 The Gitea Authors. All rights reserved.
  3. // Use of this source code is governed by a MIT-style
  4. // license that can be found in the LICENSE file.
  5. package repo
  6. import (
  7. "fmt"
  8. "io"
  9. "path"
  10. "strings"
  11. "code.gitea.io/gitea/modules/base"
  12. "code.gitea.io/gitea/modules/context"
  13. "code.gitea.io/gitea/modules/git"
  14. "code.gitea.io/gitea/modules/lfs"
  15. "code.gitea.io/gitea/modules/log"
  16. )
  17. // ServeData download file from io.Reader
  18. func ServeData(ctx *context.Context, name string, reader io.Reader) error {
  19. buf := make([]byte, 1024)
  20. n, _ := reader.Read(buf)
  21. if n >= 0 {
  22. buf = buf[:n]
  23. }
  24. ctx.Resp.Header().Set("Cache-Control", "public,max-age=86400")
  25. name = path.Base(name)
  26. // Google Chrome dislike commas in filenames, so let's change it to a space
  27. name = strings.Replace(name, ",", " ", -1)
  28. if base.IsTextFile(buf) || ctx.QueryBool("render") {
  29. ctx.Resp.Header().Set("Content-Type", "text/plain; charset=utf-8")
  30. } else if base.IsImageFile(buf) || base.IsPDFFile(buf) {
  31. ctx.Resp.Header().Set("Content-Disposition", fmt.Sprintf(`inline; filename="%s"`, name))
  32. } else {
  33. ctx.Resp.Header().Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"`, name))
  34. }
  35. _, err := ctx.Resp.Write(buf)
  36. if err != nil {
  37. return err
  38. }
  39. _, err = io.Copy(ctx.Resp, reader)
  40. return err
  41. }
  42. // ServeBlob download a git.Blob
  43. func ServeBlob(ctx *context.Context, blob *git.Blob) error {
  44. dataRc, err := blob.DataAsync()
  45. if err != nil {
  46. return err
  47. }
  48. defer func() {
  49. if err = dataRc.Close(); err != nil {
  50. log.Error("ServeBlob: Close: %v", err)
  51. }
  52. }()
  53. return ServeData(ctx, ctx.Repo.TreePath, dataRc)
  54. }
  55. // ServeBlobOrLFS download a git.Blob redirecting to LFS if necessary
  56. func ServeBlobOrLFS(ctx *context.Context, blob *git.Blob) error {
  57. dataRc, err := blob.DataAsync()
  58. if err != nil {
  59. return err
  60. }
  61. defer func() {
  62. if err = dataRc.Close(); err != nil {
  63. log.Error("ServeBlobOrLFS: Close: %v", err)
  64. }
  65. }()
  66. if meta, _ := lfs.ReadPointerFile(dataRc); meta != nil {
  67. meta, _ = ctx.Repo.Repository.GetLFSMetaObjectByOid(meta.Oid)
  68. if meta == nil {
  69. return ServeBlob(ctx, blob)
  70. }
  71. lfsDataRc, err := lfs.ReadMetaObject(meta)
  72. if err != nil {
  73. return err
  74. }
  75. return ServeData(ctx, ctx.Repo.TreePath, lfsDataRc)
  76. }
  77. return ServeBlob(ctx, blob)
  78. }
  79. // SingleDownload download a file by repos path
  80. func SingleDownload(ctx *context.Context) {
  81. blob, err := ctx.Repo.Commit.GetBlobByPath(ctx.Repo.TreePath)
  82. if err != nil {
  83. if git.IsErrNotExist(err) {
  84. ctx.NotFound("GetBlobByPath", nil)
  85. } else {
  86. ctx.ServerError("GetBlobByPath", err)
  87. }
  88. return
  89. }
  90. if err = ServeBlob(ctx, blob); err != nil {
  91. ctx.ServerError("ServeBlob", err)
  92. }
  93. }
  94. // SingleDownloadOrLFS download a file by repos path redirecting to LFS if necessary
  95. func SingleDownloadOrLFS(ctx *context.Context) {
  96. blob, err := ctx.Repo.Commit.GetBlobByPath(ctx.Repo.TreePath)
  97. if err != nil {
  98. if git.IsErrNotExist(err) {
  99. ctx.NotFound("GetBlobByPath", nil)
  100. } else {
  101. ctx.ServerError("GetBlobByPath", err)
  102. }
  103. return
  104. }
  105. if err = ServeBlobOrLFS(ctx, blob); err != nil {
  106. ctx.ServerError("ServeBlobOrLFS", err)
  107. }
  108. }
  109. // DownloadByID download a file by sha1 ID
  110. func DownloadByID(ctx *context.Context) {
  111. blob, err := ctx.Repo.GitRepo.GetBlob(ctx.Params("sha"))
  112. if err != nil {
  113. if git.IsErrNotExist(err) {
  114. ctx.NotFound("GetBlob", nil)
  115. } else {
  116. ctx.ServerError("GetBlob", err)
  117. }
  118. return
  119. }
  120. if err = ServeBlob(ctx, blob); err != nil {
  121. ctx.ServerError("ServeBlob", err)
  122. }
  123. }
  124. // DownloadByIDOrLFS download a file by sha1 ID taking account of LFS
  125. func DownloadByIDOrLFS(ctx *context.Context) {
  126. blob, err := ctx.Repo.GitRepo.GetBlob(ctx.Params("sha"))
  127. if err != nil {
  128. if git.IsErrNotExist(err) {
  129. ctx.NotFound("GetBlob", nil)
  130. } else {
  131. ctx.ServerError("GetBlob", err)
  132. }
  133. return
  134. }
  135. if err = ServeBlobOrLFS(ctx, blob); err != nil {
  136. ctx.ServerError("ServeBlob", err)
  137. }
  138. }