mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-26 09:04:38 +01:00 
			
		
		
		
	Fix object storage path handling (#27024)
Object storage path rules: * No single `/` or `.`, use empty string for root path * Need to use trailing `/` for a list prefix to distinguish a "dir/"
This commit is contained in:
		
							parent
							
								
									7d56459c6c
								
							
						
					
					
						commit
						8ecdc93f8b
					
				| @ -136,9 +136,18 @@ func NewMinioStorage(ctx context.Context, cfg *setting.Storage) (ObjectStorage, | ||||
| } | ||||
| 
 | ||||
| func (m *MinioStorage) buildMinioPath(p string) string { | ||||
| 	p = util.PathJoinRelX(m.basePath, p) | ||||
| 	p = strings.TrimPrefix(util.PathJoinRelX(m.basePath, p), "/") // object store doesn't use slash for root path | ||||
| 	if p == "." { | ||||
| 		p = "" // minio doesn't use dot as relative path | ||||
| 		p = "" // object store doesn't use dot as relative path | ||||
| 	} | ||||
| 	return p | ||||
| } | ||||
| 
 | ||||
| func (m *MinioStorage) buildMinioDirPrefix(p string) string { | ||||
| 	// ending slash is required for avoiding matching like "foo/" and "foobar/" with prefix "foo" | ||||
| 	p = m.buildMinioPath(p) + "/" | ||||
| 	if p == "/" { | ||||
| 		p = "" // object store doesn't use slash for root path | ||||
| 	} | ||||
| 	return p | ||||
| } | ||||
| @ -237,20 +246,11 @@ func (m *MinioStorage) URL(path, name string) (*url.URL, error) { | ||||
| // IterateObjects iterates across the objects in the miniostorage | ||||
| func (m *MinioStorage) IterateObjects(dirName string, fn func(path string, obj Object) error) error { | ||||
| 	opts := minio.GetObjectOptions{} | ||||
| 	lobjectCtx, cancel := context.WithCancel(m.ctx) | ||||
| 	defer cancel() | ||||
| 
 | ||||
| 	basePath := m.basePath | ||||
| 	if dirName != "" { | ||||
| 		// ending slash is required for avoiding matching like "foo/" and "foobar/" with prefix "foo" | ||||
| 		basePath = m.buildMinioPath(dirName) + "/" | ||||
| 	} | ||||
| 
 | ||||
| 	for mObjInfo := range m.client.ListObjects(lobjectCtx, m.bucket, minio.ListObjectsOptions{ | ||||
| 		Prefix:    basePath, | ||||
| 	for mObjInfo := range m.client.ListObjects(m.ctx, m.bucket, minio.ListObjectsOptions{ | ||||
| 		Prefix:    m.buildMinioDirPrefix(dirName), | ||||
| 		Recursive: true, | ||||
| 	}) { | ||||
| 		object, err := m.client.GetObject(lobjectCtx, m.bucket, mObjInfo.Key, opts) | ||||
| 		object, err := m.client.GetObject(m.ctx, m.bucket, mObjInfo.Key, opts) | ||||
| 		if err != nil { | ||||
| 			return convertMinioErr(err) | ||||
| 		} | ||||
|  | ||||
| @ -31,6 +31,40 @@ func TestMinioStorageIterator(t *testing.T) { | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func TestMinioStoragePath(t *testing.T) { | ||||
| 	m := &MinioStorage{basePath: ""} | ||||
| 	assert.Equal(t, "", m.buildMinioPath("/")) | ||||
| 	assert.Equal(t, "", m.buildMinioPath(".")) | ||||
| 	assert.Equal(t, "a", m.buildMinioPath("/a")) | ||||
| 	assert.Equal(t, "a/b", m.buildMinioPath("/a/b/")) | ||||
| 	assert.Equal(t, "", m.buildMinioDirPrefix("")) | ||||
| 	assert.Equal(t, "a/", m.buildMinioDirPrefix("/a/")) | ||||
| 
 | ||||
| 	m = &MinioStorage{basePath: "/"} | ||||
| 	assert.Equal(t, "", m.buildMinioPath("/")) | ||||
| 	assert.Equal(t, "", m.buildMinioPath(".")) | ||||
| 	assert.Equal(t, "a", m.buildMinioPath("/a")) | ||||
| 	assert.Equal(t, "a/b", m.buildMinioPath("/a/b/")) | ||||
| 	assert.Equal(t, "", m.buildMinioDirPrefix("")) | ||||
| 	assert.Equal(t, "a/", m.buildMinioDirPrefix("/a/")) | ||||
| 
 | ||||
| 	m = &MinioStorage{basePath: "/base"} | ||||
| 	assert.Equal(t, "base", m.buildMinioPath("/")) | ||||
| 	assert.Equal(t, "base", m.buildMinioPath(".")) | ||||
| 	assert.Equal(t, "base/a", m.buildMinioPath("/a")) | ||||
| 	assert.Equal(t, "base/a/b", m.buildMinioPath("/a/b/")) | ||||
| 	assert.Equal(t, "base/", m.buildMinioDirPrefix("")) | ||||
| 	assert.Equal(t, "base/a/", m.buildMinioDirPrefix("/a/")) | ||||
| 
 | ||||
| 	m = &MinioStorage{basePath: "/base/"} | ||||
| 	assert.Equal(t, "base", m.buildMinioPath("/")) | ||||
| 	assert.Equal(t, "base", m.buildMinioPath(".")) | ||||
| 	assert.Equal(t, "base/a", m.buildMinioPath("/a")) | ||||
| 	assert.Equal(t, "base/a/b", m.buildMinioPath("/a/b/")) | ||||
| 	assert.Equal(t, "base/", m.buildMinioDirPrefix("")) | ||||
| 	assert.Equal(t, "base/a/", m.buildMinioDirPrefix("/a/")) | ||||
| } | ||||
| 
 | ||||
| func TestS3StorageBadRequest(t *testing.T) { | ||||
| 	if os.Getenv("CI") == "" { | ||||
| 		t.Skip("S3Storage not present outside of CI") | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user