2014-04-15 20:27:29 +04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2018-11-18 21:45:40 +03:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2014-04-15 20:27:29 +04:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package repo
import (
2021-12-10 04:27:50 +03:00
"code.gitea.io/gitea/models"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/context"
2019-03-27 12:33:00 +03:00
"code.gitea.io/gitea/modules/git"
2021-04-12 17:49:26 +03:00
"code.gitea.io/gitea/modules/httpcache"
2019-02-12 18:09:43 +03:00
"code.gitea.io/gitea/modules/lfs"
2019-06-12 22:41:28 +03:00
"code.gitea.io/gitea/modules/log"
2021-08-21 21:22:06 +03:00
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/storage"
2021-06-09 02:33:54 +03:00
"code.gitea.io/gitea/routers/common"
2014-04-15 20:27:29 +04:00
)
2019-02-12 18:09:43 +03:00
// ServeBlobOrLFS download a git.Blob redirecting to LFS if necessary
func ServeBlobOrLFS ( ctx * context . Context , blob * git . Blob ) error {
2021-04-12 17:49:26 +03:00
if httpcache . HandleGenericETagCache ( ctx . Req , ctx . Resp , ` " ` + blob . ID . String ( ) + ` " ` ) {
return nil
}
2019-02-12 18:09:43 +03:00
dataRc , err := blob . DataAsync ( )
if err != nil {
return err
}
2021-05-10 04:27:03 +03:00
closed := false
2019-06-12 22:41:28 +03:00
defer func ( ) {
2021-05-10 04:27:03 +03:00
if closed {
return
}
2019-06-12 22:41:28 +03:00
if err = dataRc . Close ( ) ; err != nil {
log . Error ( "ServeBlobOrLFS: Close: %v" , err )
}
} ( )
2019-02-12 18:09:43 +03:00
2021-04-09 01:25:57 +03:00
pointer , _ := lfs . ReadPointer ( dataRc )
if pointer . IsValid ( ) {
2021-12-10 04:27:50 +03:00
meta , _ := models . GetLFSMetaObjectByOid ( ctx . Repo . Repository . ID , pointer . Oid )
2019-02-12 18:09:43 +03:00
if meta == nil {
2021-05-10 04:27:03 +03:00
if err = dataRc . Close ( ) ; err != nil {
log . Error ( "ServeBlobOrLFS: Close: %v" , err )
}
closed = true
2021-06-09 02:33:54 +03:00
return common . ServeBlob ( ctx , blob )
2019-02-12 18:09:43 +03:00
}
2021-04-12 17:49:26 +03:00
if httpcache . HandleGenericETagCache ( ctx . Req , ctx . Resp , ` " ` + pointer . Oid + ` " ` ) {
return nil
}
2021-08-21 21:22:06 +03:00
if setting . LFS . ServeDirect {
2022-01-20 20:46:10 +03:00
// If we have a signed url (S3, object storage), redirect to this directly.
2021-08-21 21:22:06 +03:00
u , err := storage . LFS . URL ( pointer . RelativePath ( ) , blob . Name ( ) )
if u != nil && err == nil {
ctx . Redirect ( u . String ( ) )
return nil
}
}
2021-04-09 01:25:57 +03:00
lfsDataRc , err := lfs . ReadMetaObject ( meta . Pointer )
2019-02-12 18:09:43 +03:00
if err != nil {
return err
}
2019-10-16 08:55:31 +03:00
defer func ( ) {
if err = lfsDataRc . Close ( ) ; err != nil {
log . Error ( "ServeBlobOrLFS: Close: %v" , err )
}
} ( )
2021-06-09 02:33:54 +03:00
return common . ServeData ( ctx , ctx . Repo . TreePath , meta . Size , lfsDataRc )
2019-02-12 18:09:43 +03:00
}
2021-05-10 04:27:03 +03:00
if err = dataRc . Close ( ) ; err != nil {
log . Error ( "ServeBlobOrLFS: Close: %v" , err )
}
closed = true
2019-02-12 18:09:43 +03:00
2021-06-09 02:33:54 +03:00
return common . ServeBlob ( ctx , blob )
2019-02-12 18:09:43 +03:00
}
2016-11-24 10:04:31 +03:00
// SingleDownload download a file by repos path
2016-03-11 19:56:52 +03:00
func SingleDownload ( ctx * context . Context ) {
2016-08-25 07:35:03 +03:00
blob , err := ctx . Repo . Commit . GetBlobByPath ( ctx . Repo . TreePath )
2014-11-17 05:32:26 +03:00
if err != nil {
2015-12-10 04:46:05 +03:00
if git . IsErrNotExist ( err ) {
2018-01-11 00:34:17 +03:00
ctx . NotFound ( "GetBlobByPath" , nil )
2014-11-17 05:32:26 +03:00
} else {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "GetBlobByPath" , err )
2014-11-17 05:32:26 +03:00
}
return
}
2021-06-09 02:33:54 +03:00
if err = common . ServeBlob ( ctx , blob ) ; err != nil {
2018-01-11 00:34:17 +03:00
ctx . ServerError ( "ServeBlob" , err )
2014-11-17 05:32:26 +03:00
}
2014-04-15 20:27:29 +04:00
}
2018-11-18 21:45:40 +03:00
2019-02-12 18:09:43 +03:00
// SingleDownloadOrLFS download a file by repos path redirecting to LFS if necessary
func SingleDownloadOrLFS ( ctx * context . Context ) {
blob , err := ctx . Repo . Commit . GetBlobByPath ( ctx . Repo . TreePath )
if err != nil {
if git . IsErrNotExist ( err ) {
ctx . NotFound ( "GetBlobByPath" , nil )
} else {
ctx . ServerError ( "GetBlobByPath" , err )
}
return
}
if err = ServeBlobOrLFS ( ctx , blob ) ; err != nil {
ctx . ServerError ( "ServeBlobOrLFS" , err )
}
}
2018-11-18 21:45:40 +03:00
// DownloadByID download a file by sha1 ID
func DownloadByID ( ctx * context . Context ) {
blob , err := ctx . Repo . GitRepo . GetBlob ( ctx . Params ( "sha" ) )
if err != nil {
if git . IsErrNotExist ( err ) {
ctx . NotFound ( "GetBlob" , nil )
} else {
ctx . ServerError ( "GetBlob" , err )
}
return
}
2021-06-09 02:33:54 +03:00
if err = common . ServeBlob ( ctx , blob ) ; err != nil {
2018-11-18 21:45:40 +03:00
ctx . ServerError ( "ServeBlob" , err )
}
}
2019-02-12 18:09:43 +03:00
// DownloadByIDOrLFS download a file by sha1 ID taking account of LFS
func DownloadByIDOrLFS ( ctx * context . Context ) {
blob , err := ctx . Repo . GitRepo . GetBlob ( ctx . Params ( "sha" ) )
if err != nil {
if git . IsErrNotExist ( err ) {
ctx . NotFound ( "GetBlob" , nil )
} else {
ctx . ServerError ( "GetBlob" , err )
}
return
}
if err = ServeBlobOrLFS ( ctx , blob ) ; err != nil {
ctx . ServerError ( "ServeBlob" , err )
}
}