mirror of
https://github.com/minio/minio.git
synced 2025-04-02 19:00:38 -04:00
ignore if-unmodified-since header if if-match is set (#20326)
This commit is contained in:
parent
d7ef6315ae
commit
69258d5945
@ -268,6 +268,14 @@ func checkPreconditions(ctx context.Context, w http.ResponseWriter, r *http.Requ
|
|||||||
ifNoneMatchETagHeader := r.Header.Get(xhttp.IfNoneMatch)
|
ifNoneMatchETagHeader := r.Header.Get(xhttp.IfNoneMatch)
|
||||||
if ifNoneMatchETagHeader != "" {
|
if ifNoneMatchETagHeader != "" {
|
||||||
if isETagEqual(objInfo.ETag, ifNoneMatchETagHeader) {
|
if isETagEqual(objInfo.ETag, ifNoneMatchETagHeader) {
|
||||||
|
// Do not care If-Modified-Since, Because:
|
||||||
|
// 1. If If-Modified-Since condition evaluates to true.
|
||||||
|
// If both of the If-None-Match and If-Modified-Since headers are present in the request as follows:
|
||||||
|
// If-None-Match condition evaluates to false , and;
|
||||||
|
// If-Modified-Since condition evaluates to true ;
|
||||||
|
// Then Amazon S3 returns the 304 Not Modified response code.
|
||||||
|
// 2. If If-Modified-Since condition evaluates to false, The following `ifModifiedSinceHeader` judgment will also return 304
|
||||||
|
|
||||||
// If the object ETag matches with the specified ETag.
|
// If the object ETag matches with the specified ETag.
|
||||||
writeHeadersPrecondition(w, objInfo)
|
writeHeadersPrecondition(w, objInfo)
|
||||||
w.WriteHeader(http.StatusNotModified)
|
w.WriteHeader(http.StatusNotModified)
|
||||||
@ -304,7 +312,7 @@ func checkPreconditions(ctx context.Context, w http.ResponseWriter, r *http.Requ
|
|||||||
// If-Unmodified-Since : Return the object only if it has not been modified since the specified
|
// If-Unmodified-Since : Return the object only if it has not been modified since the specified
|
||||||
// time, otherwise return a 412 (precondition failed).
|
// time, otherwise return a 412 (precondition failed).
|
||||||
ifUnmodifiedSinceHeader := r.Header.Get(xhttp.IfUnmodifiedSince)
|
ifUnmodifiedSinceHeader := r.Header.Get(xhttp.IfUnmodifiedSince)
|
||||||
if ifUnmodifiedSinceHeader != "" {
|
if ifUnmodifiedSinceHeader != "" && ifMatchETagHeader == "" {
|
||||||
if givenTime, err := amztime.ParseHeader(ifUnmodifiedSinceHeader); err == nil {
|
if givenTime, err := amztime.ParseHeader(ifUnmodifiedSinceHeader); err == nil {
|
||||||
if ifModifiedSince(objInfo.ModTime, givenTime) {
|
if ifModifiedSince(objInfo.ModTime, givenTime) {
|
||||||
// If the object is modified since the specified time.
|
// If the object is modified since the specified time.
|
||||||
|
@ -18,7 +18,14 @@
|
|||||||
package cmd
|
package cmd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
|
"context"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
"testing"
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
xhttp "github.com/minio/minio/internal/http"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Tests - canonicalizeETag()
|
// Tests - canonicalizeETag()
|
||||||
@ -51,3 +58,125 @@ func TestCanonicalizeETag(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Tests - CheckPreconditions()
|
||||||
|
func TestCheckPreconditions(t *testing.T) {
|
||||||
|
objModTime := time.Date(2024, time.August, 26, 0o2, 0o1, 0o1, 0, time.UTC)
|
||||||
|
objInfo := ObjectInfo{ETag: "aa", ModTime: objModTime}
|
||||||
|
testCases := []struct {
|
||||||
|
name string
|
||||||
|
ifMatch string
|
||||||
|
ifNoneMatch string
|
||||||
|
ifModifiedSince string
|
||||||
|
ifUnmodifiedSince string
|
||||||
|
objInfo ObjectInfo
|
||||||
|
expectedFlag bool
|
||||||
|
expectedCode int
|
||||||
|
}{
|
||||||
|
// If-None-Match(false) and If-Modified-Since(true)
|
||||||
|
{
|
||||||
|
name: "If-None-Match1",
|
||||||
|
ifNoneMatch: "aa",
|
||||||
|
ifModifiedSince: "Sun, 26 Aug 2024 02:01:00 GMT",
|
||||||
|
objInfo: objInfo,
|
||||||
|
expectedFlag: true,
|
||||||
|
expectedCode: 304,
|
||||||
|
},
|
||||||
|
// If-Modified-Since(false)
|
||||||
|
{
|
||||||
|
name: "If-None-Match2",
|
||||||
|
ifNoneMatch: "aaa",
|
||||||
|
ifModifiedSince: "Sun, 26 Aug 2024 02:01:01 GMT",
|
||||||
|
objInfo: objInfo,
|
||||||
|
expectedFlag: true,
|
||||||
|
expectedCode: 304,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "If-None-Match3",
|
||||||
|
ifNoneMatch: "aaa",
|
||||||
|
ifModifiedSince: "Sun, 26 Aug 2024 02:01:02 GMT",
|
||||||
|
objInfo: objInfo,
|
||||||
|
expectedFlag: true,
|
||||||
|
expectedCode: 304,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
recorder := httptest.NewRecorder()
|
||||||
|
request := httptest.NewRequest(http.MethodHead, "/bucket/a", bytes.NewReader([]byte{}))
|
||||||
|
request.Header.Set(xhttp.IfNoneMatch, tc.ifNoneMatch)
|
||||||
|
request.Header.Set(xhttp.IfModifiedSince, tc.ifModifiedSince)
|
||||||
|
request.Header.Set(xhttp.IfMatch, tc.ifMatch)
|
||||||
|
request.Header.Set(xhttp.IfUnmodifiedSince, tc.ifUnmodifiedSince)
|
||||||
|
actualFlag := checkPreconditions(context.Background(), recorder, request, tc.objInfo, ObjectOptions{})
|
||||||
|
if tc.expectedFlag != actualFlag {
|
||||||
|
t.Errorf("test: %s, got flag: %v, want: %v", tc.name, actualFlag, tc.expectedFlag)
|
||||||
|
}
|
||||||
|
if tc.expectedCode != recorder.Code {
|
||||||
|
t.Errorf("test: %s, got code: %d, want: %d", tc.name, recorder.Code, tc.expectedCode)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
testCases = []struct {
|
||||||
|
name string
|
||||||
|
ifMatch string
|
||||||
|
ifNoneMatch string
|
||||||
|
ifModifiedSince string
|
||||||
|
ifUnmodifiedSince string
|
||||||
|
objInfo ObjectInfo
|
||||||
|
expectedFlag bool
|
||||||
|
expectedCode int
|
||||||
|
}{
|
||||||
|
// If-Match(true) and If-Unmodified-Since(false)
|
||||||
|
{
|
||||||
|
name: "If-Match1",
|
||||||
|
ifMatch: "aa",
|
||||||
|
ifUnmodifiedSince: "Sun, 26 Aug 2024 02:01:00 GMT",
|
||||||
|
objInfo: objInfo,
|
||||||
|
expectedFlag: false,
|
||||||
|
expectedCode: 200,
|
||||||
|
},
|
||||||
|
// If-Unmodified-Since(true)
|
||||||
|
{
|
||||||
|
name: "If-Match2",
|
||||||
|
ifMatch: "aa",
|
||||||
|
ifUnmodifiedSince: "Sun, 26 Aug 2024 02:01:01 GMT",
|
||||||
|
objInfo: objInfo,
|
||||||
|
expectedFlag: false,
|
||||||
|
expectedCode: 200,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "If-Match3",
|
||||||
|
ifMatch: "aa",
|
||||||
|
ifUnmodifiedSince: "Sun, 26 Aug 2024 02:01:02 GMT",
|
||||||
|
objInfo: objInfo,
|
||||||
|
expectedFlag: false,
|
||||||
|
expectedCode: 200,
|
||||||
|
},
|
||||||
|
// If-Match(true)
|
||||||
|
{
|
||||||
|
name: "If-Match4",
|
||||||
|
ifMatch: "aa",
|
||||||
|
objInfo: objInfo,
|
||||||
|
expectedFlag: false,
|
||||||
|
expectedCode: 200,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tc := range testCases {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
recorder := httptest.NewRecorder()
|
||||||
|
request := httptest.NewRequest(http.MethodHead, "/bucket/a", bytes.NewReader([]byte{}))
|
||||||
|
request.Header.Set(xhttp.IfNoneMatch, tc.ifNoneMatch)
|
||||||
|
request.Header.Set(xhttp.IfModifiedSince, tc.ifModifiedSince)
|
||||||
|
request.Header.Set(xhttp.IfMatch, tc.ifMatch)
|
||||||
|
request.Header.Set(xhttp.IfUnmodifiedSince, tc.ifUnmodifiedSince)
|
||||||
|
actualFlag := checkPreconditions(context.Background(), recorder, request, tc.objInfo, ObjectOptions{})
|
||||||
|
if tc.expectedFlag != actualFlag {
|
||||||
|
t.Errorf("test: %s, got flag: %v, want: %v", tc.name, actualFlag, tc.expectedFlag)
|
||||||
|
}
|
||||||
|
if tc.expectedCode != recorder.Code {
|
||||||
|
t.Errorf("test: %s, got code: %d, want: %d", tc.name, recorder.Code, tc.expectedCode)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user