| 
									
										
										
										
											2021-04-19 03:41:13 +08:00
										 |  |  | // Copyright (c) 2015-2021 MinIO, Inc.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This file is part of MinIO Object Storage stack
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This program is free software: you can redistribute it and/or modify
 | 
					
						
							|  |  |  | // it under the terms of the GNU Affero General Public License as published by
 | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or
 | 
					
						
							|  |  |  | // (at your option) any later version.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This program is distributed in the hope that it will be useful
 | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
					
						
							|  |  |  | // GNU Affero General Public License for more details.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // You should have received a copy of the GNU Affero General Public License
 | 
					
						
							|  |  |  | // along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							| 
									
										
										
										
											2019-08-10 08:09:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | package cmd | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"net/http" | 
					
						
							|  |  |  | 	"reflect" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestGetCacheControlOpts(t *testing.T) { | 
					
						
							|  |  |  | 	expiry, _ := time.Parse(http.TimeFormat, "Wed, 21 Oct 2015 07:28:00 GMT") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	testCases := []struct { | 
					
						
							|  |  |  | 		cacheControlHeaderVal string | 
					
						
							|  |  |  | 		expiryHeaderVal       time.Time | 
					
						
							| 
									
										
										
										
											2020-02-23 21:33:39 +08:00
										 |  |  | 		expectedCacheControl  *cacheControl | 
					
						
							| 
									
										
										
										
											2019-08-10 08:09:08 +08:00
										 |  |  | 		expectedErr           bool | 
					
						
							|  |  |  | 	}{ | 
					
						
							| 
									
										
										
										
											2020-02-23 21:33:39 +08:00
										 |  |  | 		{"", timeSentinel, nil, false}, | 
					
						
							|  |  |  | 		{"max-age=2592000, public", timeSentinel, &cacheControl{maxAge: 2592000, sMaxAge: 0, minFresh: 0, expiry: time.Time{}}, false}, | 
					
						
							|  |  |  | 		{"max-age=2592000, no-store", timeSentinel, &cacheControl{maxAge: 2592000, sMaxAge: 0, noStore: true, minFresh: 0, expiry: time.Time{}}, false}, | 
					
						
							|  |  |  | 		{"must-revalidate, max-age=600", timeSentinel, &cacheControl{maxAge: 600, sMaxAge: 0, minFresh: 0, expiry: time.Time{}}, false}, | 
					
						
							|  |  |  | 		{"s-maxAge=2500, max-age=600", timeSentinel, &cacheControl{maxAge: 600, sMaxAge: 2500, minFresh: 0, expiry: time.Time{}}, false}, | 
					
						
							| 
									
										
										
										
											2022-01-03 01:15:06 +08:00
										 |  |  | 		{"s-maxAge=2500, max-age=600", expiry, &cacheControl{maxAge: 600, sMaxAge: 2500, minFresh: 0, expiry: time.Date(2015, time.October, 21, 0o7, 28, 0o0, 0o0, time.UTC)}, false}, | 
					
						
							| 
									
										
										
										
											2020-02-23 21:33:39 +08:00
										 |  |  | 		{"s-maxAge=2500, max-age=600s", timeSentinel, &cacheControl{maxAge: 600, sMaxAge: 2500, minFresh: 0, expiry: time.Time{}}, true}, | 
					
						
							| 
									
										
										
										
											2019-08-10 08:09:08 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-12 05:26:18 +08:00
										 |  |  | 	for _, testCase := range testCases { | 
					
						
							|  |  |  | 		t.Run("", func(t *testing.T) { | 
					
						
							|  |  |  | 			m := make(map[string]string) | 
					
						
							|  |  |  | 			m["cache-control"] = testCase.cacheControlHeaderVal | 
					
						
							| 
									
										
										
										
											2020-08-25 03:11:20 +08:00
										 |  |  | 			if !testCase.expiryHeaderVal.Equal(timeSentinel) { | 
					
						
							| 
									
										
										
										
											2019-12-12 05:26:18 +08:00
										 |  |  | 				m["expires"] = testCase.expiryHeaderVal.String() | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			c := cacheControlOpts(ObjectInfo{UserDefined: m, Expires: testCase.expiryHeaderVal}) | 
					
						
							| 
									
										
										
										
											2020-02-23 21:33:39 +08:00
										 |  |  | 			if testCase.expectedErr && (c != nil) { | 
					
						
							| 
									
										
										
										
											2019-12-12 05:26:18 +08:00
										 |  |  | 				t.Errorf("expected err, got <nil>") | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if !testCase.expectedErr && !reflect.DeepEqual(c, testCase.expectedCacheControl) { | 
					
						
							|  |  |  | 				t.Errorf("expected %v, got %v", testCase.expectedCacheControl, c) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2019-08-10 08:09:08 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-02-04 11:40:01 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestIsMetadataSame(t *testing.T) { | 
					
						
							|  |  |  | 	testCases := []struct { | 
					
						
							|  |  |  | 		m1       map[string]string | 
					
						
							|  |  |  | 		m2       map[string]string | 
					
						
							|  |  |  | 		expected bool | 
					
						
							|  |  |  | 	}{ | 
					
						
							|  |  |  | 		{nil, nil, true}, | 
					
						
							|  |  |  | 		{nil, map[string]string{}, false}, | 
					
						
							|  |  |  | 		{map[string]string{"k": "v"}, map[string]string{"k": "v"}, true}, | 
					
						
							|  |  |  | 		{map[string]string{"k": "v"}, map[string]string{"a": "b"}, false}, | 
					
						
							|  |  |  | 		{map[string]string{"k1": "v1", "k2": "v2"}, map[string]string{"k1": "v1", "k2": "v1"}, false}, | 
					
						
							|  |  |  | 		{map[string]string{"k1": "v1", "k2": "v2"}, map[string]string{"k1": "v1", "k2": "v2"}, true}, | 
					
						
							|  |  |  | 		{map[string]string{"K1": "v1", "k2": "v2"}, map[string]string{"k1": "v1", "k2": "v2"}, false}, | 
					
						
							|  |  |  | 		{map[string]string{"k1": "v1", "k2": "v2", "k3": "v3"}, map[string]string{"k1": "v1", "k2": "v2"}, false}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for i, testCase := range testCases { | 
					
						
							|  |  |  | 		actual := isMetadataSame(testCase.m1, testCase.m2) | 
					
						
							|  |  |  | 		if testCase.expected != actual { | 
					
						
							|  |  |  | 			t.Errorf("test %d expected %v, got %v", i, testCase.expected, actual) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-02-23 21:33:39 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func TestNewFileScorer(t *testing.T) { | 
					
						
							|  |  |  | 	fs, err := newFileScorer(1000, time.Now().Unix(), 10) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if len(fs.fileNames()) != 0 { | 
					
						
							|  |  |  | 		t.Fatal("non zero files??") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	now := time.Now() | 
					
						
							|  |  |  | 	fs.addFile("recent", now.Add(-time.Minute), 1000, 10) | 
					
						
							|  |  |  | 	fs.addFile("older", now.Add(-time.Hour), 1000, 10) | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(fs.fileNames(), []string{"older"}) { | 
					
						
							|  |  |  | 		t.Fatal("unexpected file list", fs.queueString()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fs.reset() | 
					
						
							|  |  |  | 	fs.addFile("bigger", now.Add(-time.Minute), 2000, 10) | 
					
						
							|  |  |  | 	fs.addFile("recent", now.Add(-time.Minute), 1000, 10) | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(fs.fileNames(), []string{"bigger"}) { | 
					
						
							|  |  |  | 		t.Fatal("unexpected file list", fs.queueString()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fs.reset() | 
					
						
							|  |  |  | 	fs.addFile("less", now.Add(-time.Minute), 1000, 5) | 
					
						
							|  |  |  | 	fs.addFile("recent", now.Add(-time.Minute), 1000, 10) | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(fs.fileNames(), []string{"less"}) { | 
					
						
							|  |  |  | 		t.Fatal("unexpected file list", fs.queueString()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fs.reset() | 
					
						
							|  |  |  | 	fs.addFile("small", now.Add(-time.Minute), 200, 10) | 
					
						
							|  |  |  | 	fs.addFile("medium", now.Add(-time.Minute), 300, 10) | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(fs.fileNames(), []string{"medium", "small"}) { | 
					
						
							|  |  |  | 		t.Fatal("unexpected file list", fs.queueString()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fs.addFile("large", now.Add(-time.Minute), 700, 10) | 
					
						
							|  |  |  | 	fs.addFile("xsmol", now.Add(-time.Minute), 7, 10) | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(fs.fileNames(), []string{"large", "medium"}) { | 
					
						
							|  |  |  | 		t.Fatal("unexpected file list", fs.queueString()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fs.reset() | 
					
						
							|  |  |  | 	fs.addFile("less", now.Add(-time.Minute), 500, 5) | 
					
						
							|  |  |  | 	fs.addFile("recent", now.Add(-time.Minute), 500, 10) | 
					
						
							|  |  |  | 	if !fs.adjustSaveBytes(-500) { | 
					
						
							|  |  |  | 		t.Fatal("we should still need more bytes, got false") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// We should only need 500 bytes now.
 | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(fs.fileNames(), []string{"less"}) { | 
					
						
							|  |  |  | 		t.Fatal("unexpected file list", fs.queueString()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if fs.adjustSaveBytes(-500) { | 
					
						
							|  |  |  | 		t.Fatal("we shouldn't need any more bytes, got true") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fs, err = newFileScorer(1000, time.Now().Unix(), 10) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		t.Fatal(err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fs.addFile("bigger", now.Add(-time.Minute), 50, 10) | 
					
						
							|  |  |  | 	// sorting should be consistent after adjusting savebytes.
 | 
					
						
							|  |  |  | 	fs.adjustSaveBytes(-800) | 
					
						
							|  |  |  | 	fs.addFile("smaller", now.Add(-time.Minute), 40, 10) | 
					
						
							|  |  |  | 	if !reflect.DeepEqual(fs.fileNames(), []string{"bigger", "smaller"}) { | 
					
						
							|  |  |  | 		t.Fatal("unexpected file list", fs.queueString()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-01-03 01:15:06 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-23 21:33:39 +08:00
										 |  |  | func TestBytesToClear(t *testing.T) { | 
					
						
							|  |  |  | 	testCases := []struct { | 
					
						
							| 
									
										
										
										
											2020-09-03 08:48:44 +08:00
										 |  |  | 		total         int64 | 
					
						
							|  |  |  | 		free          int64 | 
					
						
							|  |  |  | 		quotaPct      uint64 | 
					
						
							|  |  |  | 		watermarkLow  uint64 | 
					
						
							|  |  |  | 		watermarkHigh uint64 | 
					
						
							|  |  |  | 		expected      uint64 | 
					
						
							| 
									
										
										
										
											2020-02-23 21:33:39 +08:00
										 |  |  | 	}{ | 
					
						
							| 
									
										
										
										
											2020-09-03 08:48:44 +08:00
										 |  |  | 		{total: 1000, free: 800, quotaPct: 40, watermarkLow: 90, watermarkHigh: 90, expected: 0}, | 
					
						
							|  |  |  | 		{total: 1000, free: 200, quotaPct: 40, watermarkLow: 90, watermarkHigh: 90, expected: 400}, | 
					
						
							|  |  |  | 		{total: 1000, free: 400, quotaPct: 40, watermarkLow: 90, watermarkHigh: 90, expected: 240}, | 
					
						
							|  |  |  | 		{total: 1000, free: 600, quotaPct: 40, watermarkLow: 90, watermarkHigh: 90, expected: 40}, | 
					
						
							|  |  |  | 		{total: 1000, free: 600, quotaPct: 40, watermarkLow: 70, watermarkHigh: 70, expected: 120}, | 
					
						
							|  |  |  | 		{total: 1000, free: 1000, quotaPct: 90, watermarkLow: 70, watermarkHigh: 70, expected: 0}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// High not yet reached..
 | 
					
						
							|  |  |  | 		{total: 1000, free: 250, quotaPct: 100, watermarkLow: 50, watermarkHigh: 90, expected: 0}, | 
					
						
							|  |  |  | 		{total: 1000, free: 250, quotaPct: 100, watermarkLow: 50, watermarkHigh: 90, expected: 0}, | 
					
						
							| 
									
										
										
										
											2020-02-23 21:33:39 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	for i, tc := range testCases { | 
					
						
							| 
									
										
										
										
											2020-09-03 08:48:44 +08:00
										 |  |  | 		toClear := bytesToClear(tc.total, tc.free, tc.quotaPct, tc.watermarkLow, tc.watermarkHigh) | 
					
						
							| 
									
										
										
										
											2020-02-23 21:33:39 +08:00
										 |  |  | 		if tc.expected != toClear { | 
					
						
							|  |  |  | 			t.Errorf("test %d expected %v, got %v", i, tc.expected, toClear) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |