2020-11-13 15:51:07 +03:00
// Copyright 2020 The Gitea Authors. All rights reserved.
2022-11-27 21:20:29 +03:00
// SPDX-License-Identifier: MIT
2020-11-13 15:51:07 +03:00
2021-06-09 02:33:54 +03:00
package web
2020-11-13 15:51:07 +03:00
import (
"errors"
"fmt"
"net/http"
"os"
"path"
"strings"
2020-11-18 01:44:52 +03:00
"code.gitea.io/gitea/modules/httpcache"
2020-11-13 15:51:07 +03:00
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/storage"
2023-03-08 15:17:39 +03:00
"code.gitea.io/gitea/modules/util"
2022-01-20 14:41:25 +03:00
"code.gitea.io/gitea/modules/web/routing"
2020-11-13 15:51:07 +03:00
)
2023-10-06 16:23:14 +03:00
func storageHandler ( storageSetting * setting . Storage , prefix string , objStore storage . ObjectStorage ) http . HandlerFunc {
2022-03-23 00:02:26 +03:00
prefix = strings . Trim ( prefix , "/" )
2022-01-20 14:41:25 +03:00
funcInfo := routing . GetFuncInfo ( storageHandler , prefix )
2020-11-13 15:51:07 +03:00
2023-10-06 16:23:14 +03:00
if storageSetting . MinioConfig . ServeDirect {
2020-11-13 15:51:07 +03:00
return http . HandlerFunc ( func ( w http . ResponseWriter , req * http . Request ) {
if req . Method != "GET" && req . Method != "HEAD" {
2023-10-06 16:23:14 +03:00
http . Error ( w , http . StatusText ( http . StatusMethodNotAllowed ) , http . StatusMethodNotAllowed )
2020-11-13 15:51:07 +03:00
return
}
2022-03-23 00:02:26 +03:00
if ! strings . HasPrefix ( req . URL . Path , "/" + prefix + "/" ) {
2023-10-06 16:23:14 +03:00
http . Error ( w , http . StatusText ( http . StatusNotFound ) , http . StatusNotFound )
2020-11-13 15:51:07 +03:00
return
}
2022-01-20 14:41:25 +03:00
routing . UpdateFuncInfo ( req . Context ( ) , funcInfo )
2020-11-13 15:51:07 +03:00
2022-03-23 00:02:26 +03:00
rPath := strings . TrimPrefix ( req . URL . Path , "/" + prefix + "/" )
2023-03-21 23:02:49 +03:00
rPath = util . PathJoinRelX ( rPath )
2020-11-18 01:44:52 +03:00
Fix `missing signature key` error when pulling Docker images with `SERVE_DIRECT` enabled (#32365)
Fix #28121
I did some tests and found that the `missing signature key` error is
caused by an incorrect `Content-Type` header. Gitea correctly sets the
`Content-Type` header when serving files.
https://github.com/go-gitea/gitea/blob/348d1d0f322ca57c459acd902f54821d687ca804/routers/api/packages/container/container.go#L712-L717
However, when `SERVE_DIRECT` is enabled, the `Content-Type` header may
be set to an incorrect value by the storage service. To fix this issue,
we can use query parameters to override response header values.
https://docs.aws.amazon.com/AmazonS3/latest/API/API_GetObject.html
<img width="600px"
src="https://github.com/user-attachments/assets/f2ff90f0-f1df-46f9-9680-b8120222c555"
/>
In this PR, I introduced a new parameter to the `URL` method to support
additional parameters.
```
URL(path, name string, reqParams url.Values) (*url.URL, error)
```
---
Most S3-like services support specifying the content type when storing
objects. However, Gitea always use `application/octet-stream`.
Therefore, I believe we also need to improve the `Save` method to
support storing objects with the correct content type.
https://github.com/go-gitea/gitea/blob/b7fb20e73e63b8edc9b90c52073e248bef428fcc/modules/storage/minio.go#L214-L221
(cherry picked from commit 0690cb076bf63f71988a709f62a9c04660b51a4f)
Conflicts:
- modules/storage/azureblob.go
Dropped the change, as we do not support Azure blob storage.
- modules/storage/helper.go
Resolved by adjusting their `discardStorage` to our
`DiscardStorage`
- routers/api/actions/artifacts.go
routers/api/actions/artifactsv4.go
routers/web/repo/actions/view.go
routers/web/repo/download.go
Resolved the conflicts by manually adding the new `nil`
parameter to the `storage.Attachments.URL()` calls.
Originally conflicted due to differences in the if expression
above these calls.
2024-10-31 18:28:25 +03:00
u , err := objStore . URL ( rPath , path . Base ( rPath ) , nil )
2020-11-13 15:51:07 +03:00
if err != nil {
if os . IsNotExist ( err ) || errors . Is ( err , os . ErrNotExist ) {
log . Warn ( "Unable to find %s %s" , prefix , rPath )
2023-10-06 16:23:14 +03:00
http . Error ( w , http . StatusText ( http . StatusNotFound ) , http . StatusNotFound )
2020-11-13 15:51:07 +03:00
return
}
2023-10-06 16:23:14 +03:00
log . Error ( "Error whilst getting URL for %s %s. Error: %v" , prefix , rPath , err )
http . Error ( w , fmt . Sprintf ( "Error whilst getting URL for %s %s" , prefix , rPath ) , http . StatusInternalServerError )
2020-11-13 15:51:07 +03:00
return
}
2023-10-06 16:23:14 +03:00
http . Redirect ( w , req , u . String ( ) , http . StatusTemporaryRedirect )
2020-11-13 15:51:07 +03:00
} )
}
2023-10-06 16:23:14 +03:00
return http . HandlerFunc ( func ( w http . ResponseWriter , req * http . Request ) {
if req . Method != "GET" && req . Method != "HEAD" {
http . Error ( w , http . StatusText ( http . StatusMethodNotAllowed ) , http . StatusMethodNotAllowed )
return
}
if ! strings . HasPrefix ( req . URL . Path , "/" + prefix + "/" ) {
http . Error ( w , http . StatusText ( http . StatusNotFound ) , http . StatusNotFound )
return
}
routing . UpdateFuncInfo ( req . Context ( ) , funcInfo )
rPath := strings . TrimPrefix ( req . URL . Path , "/" + prefix + "/" )
rPath = util . PathJoinRelX ( rPath )
if rPath == "" || rPath == "." {
http . Error ( w , http . StatusText ( http . StatusNotFound ) , http . StatusNotFound )
return
}
fi , err := objStore . Stat ( rPath )
if err != nil {
if os . IsNotExist ( err ) || errors . Is ( err , os . ErrNotExist ) {
log . Warn ( "Unable to find %s %s" , prefix , rPath )
http . Error ( w , http . StatusText ( http . StatusNotFound ) , http . StatusNotFound )
return
}
log . Error ( "Error whilst opening %s %s. Error: %v" , prefix , rPath , err )
http . Error ( w , fmt . Sprintf ( "Error whilst opening %s %s" , prefix , rPath ) , http . StatusInternalServerError )
return
}
fr , err := objStore . Open ( rPath )
if err != nil {
log . Error ( "Error whilst opening %s %s. Error: %v" , prefix , rPath , err )
http . Error ( w , fmt . Sprintf ( "Error whilst opening %s %s" , prefix , rPath ) , http . StatusInternalServerError )
return
}
defer fr . Close ( )
httpcache . ServeContentWithCacheControl ( w , req , path . Base ( rPath ) , fi . ModTime ( ) , fr )
} )
2020-11-13 15:51:07 +03:00
}