mirror of
				https://github.com/go-gitea/gitea
				synced 2025-11-04 13:28:25 +00:00 
			
		
		
		
	related #16865 This PR adds an accessibility check before mounting container blobs. --------- Co-authored-by: techknowlogick <techknowlogick@gitea.io> Co-authored-by: silverwind <me@silverwind.io>
		
			
				
	
	
		
			147 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			147 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2021 The Gitea Authors. All rights reserved.
 | 
						|
// SPDX-License-Identifier: MIT
 | 
						|
 | 
						|
package packages
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"strconv"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"code.gitea.io/gitea/models/db"
 | 
						|
	"code.gitea.io/gitea/models/perm"
 | 
						|
	"code.gitea.io/gitea/models/unit"
 | 
						|
	user_model "code.gitea.io/gitea/models/user"
 | 
						|
	"code.gitea.io/gitea/modules/structs"
 | 
						|
	"code.gitea.io/gitea/modules/timeutil"
 | 
						|
	"code.gitea.io/gitea/modules/util"
 | 
						|
 | 
						|
	"xorm.io/builder"
 | 
						|
)
 | 
						|
 | 
						|
// ErrPackageBlobNotExist indicates a package blob not exist error
 | 
						|
var ErrPackageBlobNotExist = util.NewNotExistErrorf("package blob does not exist")
 | 
						|
 | 
						|
func init() {
 | 
						|
	db.RegisterModel(new(PackageBlob))
 | 
						|
}
 | 
						|
 | 
						|
// PackageBlob represents a package blob
 | 
						|
type PackageBlob struct {
 | 
						|
	ID          int64              `xorm:"pk autoincr"`
 | 
						|
	Size        int64              `xorm:"NOT NULL DEFAULT 0"`
 | 
						|
	HashMD5     string             `xorm:"hash_md5 char(32) UNIQUE(md5) INDEX NOT NULL"`
 | 
						|
	HashSHA1    string             `xorm:"hash_sha1 char(40) UNIQUE(sha1) INDEX NOT NULL"`
 | 
						|
	HashSHA256  string             `xorm:"hash_sha256 char(64) UNIQUE(sha256) INDEX NOT NULL"`
 | 
						|
	HashSHA512  string             `xorm:"hash_sha512 char(128) UNIQUE(sha512) INDEX NOT NULL"`
 | 
						|
	CreatedUnix timeutil.TimeStamp `xorm:"created INDEX NOT NULL"`
 | 
						|
}
 | 
						|
 | 
						|
// GetOrInsertBlob inserts a blob. If the blob exists already the existing blob is returned
 | 
						|
func GetOrInsertBlob(ctx context.Context, pb *PackageBlob) (*PackageBlob, bool, error) {
 | 
						|
	e := db.GetEngine(ctx)
 | 
						|
 | 
						|
	has, err := e.Get(pb)
 | 
						|
	if err != nil {
 | 
						|
		return nil, false, err
 | 
						|
	}
 | 
						|
	if has {
 | 
						|
		return pb, true, nil
 | 
						|
	}
 | 
						|
	if _, err = e.Insert(pb); err != nil {
 | 
						|
		return nil, false, err
 | 
						|
	}
 | 
						|
	return pb, false, nil
 | 
						|
}
 | 
						|
 | 
						|
// GetBlobByID gets a blob by id
 | 
						|
func GetBlobByID(ctx context.Context, blobID int64) (*PackageBlob, error) {
 | 
						|
	pb := &PackageBlob{}
 | 
						|
 | 
						|
	has, err := db.GetEngine(ctx).ID(blobID).Get(pb)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	if !has {
 | 
						|
		return nil, ErrPackageBlobNotExist
 | 
						|
	}
 | 
						|
	return pb, nil
 | 
						|
}
 | 
						|
 | 
						|
// ExistPackageBlobWithSHA returns if a package blob exists with the provided sha
 | 
						|
func ExistPackageBlobWithSHA(ctx context.Context, blobSha256 string) (bool, error) {
 | 
						|
	return db.GetEngine(ctx).Exist(&PackageBlob{
 | 
						|
		HashSHA256: blobSha256,
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
// FindExpiredUnreferencedBlobs gets all blobs without associated files older than the specific duration
 | 
						|
func FindExpiredUnreferencedBlobs(ctx context.Context, olderThan time.Duration) ([]*PackageBlob, error) {
 | 
						|
	pbs := make([]*PackageBlob, 0, 10)
 | 
						|
	return pbs, db.GetEngine(ctx).
 | 
						|
		Table("package_blob").
 | 
						|
		Join("LEFT", "package_file", "package_file.blob_id = package_blob.id").
 | 
						|
		Where("package_file.id IS NULL AND package_blob.created_unix < ?", time.Now().Add(-olderThan).Unix()).
 | 
						|
		Find(&pbs)
 | 
						|
}
 | 
						|
 | 
						|
// DeleteBlobByID deletes a blob by id
 | 
						|
func DeleteBlobByID(ctx context.Context, blobID int64) error {
 | 
						|
	_, err := db.GetEngine(ctx).ID(blobID).Delete(&PackageBlob{})
 | 
						|
	return err
 | 
						|
}
 | 
						|
 | 
						|
// GetTotalBlobSize returns the total blobs size in bytes
 | 
						|
func GetTotalBlobSize(ctx context.Context) (int64, error) {
 | 
						|
	return db.GetEngine(ctx).
 | 
						|
		SumInt(&PackageBlob{}, "size")
 | 
						|
}
 | 
						|
 | 
						|
// GetTotalUnreferencedBlobSize returns the total size of all unreferenced blobs in bytes
 | 
						|
func GetTotalUnreferencedBlobSize(ctx context.Context) (int64, error) {
 | 
						|
	return db.GetEngine(ctx).
 | 
						|
		Table("package_blob").
 | 
						|
		Join("LEFT", "package_file", "package_file.blob_id = package_blob.id").
 | 
						|
		Where("package_file.id IS NULL").
 | 
						|
		SumInt(&PackageBlob{}, "size")
 | 
						|
}
 | 
						|
 | 
						|
// IsBlobAccessibleForUser tests if the user has access to the blob
 | 
						|
func IsBlobAccessibleForUser(ctx context.Context, blobID int64, user *user_model.User) (bool, error) {
 | 
						|
	if user.IsAdmin {
 | 
						|
		return true, nil
 | 
						|
	}
 | 
						|
 | 
						|
	maxTeamAuthorize := builder.
 | 
						|
		Select("max(team.authorize)").
 | 
						|
		From("team").
 | 
						|
		InnerJoin("team_user", "team_user.team_id = team.id").
 | 
						|
		Where(builder.Eq{"team_user.uid": user.ID}.And(builder.Expr("team_user.org_id = `user`.id")))
 | 
						|
 | 
						|
	maxTeamUnitAccessMode := builder.
 | 
						|
		Select("max(team_unit.access_mode)").
 | 
						|
		From("team").
 | 
						|
		InnerJoin("team_user", "team_user.team_id = team.id").
 | 
						|
		InnerJoin("team_unit", "team_unit.team_id = team.id").
 | 
						|
		Where(builder.Eq{"team_user.uid": user.ID, "team_unit.type": unit.TypePackages}.And(builder.Expr("team_user.org_id = `user`.id")))
 | 
						|
 | 
						|
	cond := builder.Eq{"package_blob.id": blobID}.And(
 | 
						|
		// owner = user
 | 
						|
		builder.Eq{"`user`.id": user.ID}.
 | 
						|
			// user can see owner
 | 
						|
			Or(builder.Eq{"`user`.visibility": structs.VisibleTypePublic}.Or(builder.Eq{"`user`.visibility": structs.VisibleTypeLimited})).
 | 
						|
			// owner is an organization and user has access to it
 | 
						|
			Or(builder.Eq{"`user`.type": user_model.UserTypeOrganization}.
 | 
						|
				And(builder.Lte{strconv.Itoa(int(perm.AccessModeRead)): maxTeamAuthorize}.Or(builder.Lte{strconv.Itoa(int(perm.AccessModeRead)): maxTeamUnitAccessMode}))),
 | 
						|
	)
 | 
						|
 | 
						|
	return db.GetEngine(ctx).
 | 
						|
		Table("package_blob").
 | 
						|
		Join("INNER", "package_file", "package_file.blob_id = package_blob.id").
 | 
						|
		Join("INNER", "package_version", "package_version.id = package_file.version_id").
 | 
						|
		Join("INNER", "package", "package.id = package_version.package_id").
 | 
						|
		Join("INNER", "user", "`user`.id = package.owner_id").
 | 
						|
		Where(cond).
 | 
						|
		Exist(&PackageBlob{})
 | 
						|
}
 |