* paginate results * fixed deadlock * prevented breaking change * updated swagger * go fmt * fixed find topic * go mod tidy * go mod vendor with go1.13.5 * fixed repo find topics * fixed unit test * added Limit method to Engine struct; use engine variable when provided; fixed gitignore * use ItemsPerPage for default pagesize; fix GetWatchers, getOrgUsersByOrgID and GetStargazers; fix GetAllCommits headers; reverted some changed behaviors * set Page value on Home route * improved memory allocations * fixed response headers * removed logfiles * fixed import order * import order * improved swagger * added function to get models.ListOptions from context * removed pagesize diff on unit test * fixed imports * removed unnecessary struct field * fixed go fmt * scoped PR * code improvements * code improvements * go mod tidy * fixed import order * fixed commit statuses session * fixed files headers * fixed headers; added pagination for notifications * go mod tidy * go fmt * removed Private from user search options; added setting.UI.IssuePagingNum as default valeu on repo's issues list * Apply suggestions from code review Co-Authored-By: 6543 <6543@obermui.de> Co-Authored-By: zeripath <art27@cantab.net> * fixed build error * CI.restart() * fixed merge conflicts resolve * fixed conflicts resolve * improved FindTrackedTimesOptions.ToOptions() method * added backwards compatibility on ListReleases request; fixed issue tracked time ToSession * fixed build error; fixed swagger template * fixed swagger template * fixed ListReleases backwards compatibility * added page to user search route Co-authored-by: techknowlogick <matti@mdranta.net> Co-authored-by: 6543 <6543@obermui.de> Co-authored-by: zeripath <art27@cantab.net>
		
			
				
	
	
		
			382 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			382 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2017 The Gitea Authors. All rights reserved.
 | |
| // Use of this source code is governed by a MIT-style
 | |
| // license that can be found in the LICENSE file.
 | |
| 
 | |
| package models
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"strings"
 | |
| 
 | |
| 	"code.gitea.io/gitea/modules/structs"
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| 
 | |
| 	"xorm.io/builder"
 | |
| )
 | |
| 
 | |
| // RepositoryListDefaultPageSize is the default number of repositories
 | |
| // to load in memory when running administrative tasks on all (or almost
 | |
| // all) of them.
 | |
| // The number should be low enough to avoid filling up all RAM with
 | |
| // repository data...
 | |
| const RepositoryListDefaultPageSize = 64
 | |
| 
 | |
| // RepositoryList contains a list of repositories
 | |
| type RepositoryList []*Repository
 | |
| 
 | |
| func (repos RepositoryList) Len() int {
 | |
| 	return len(repos)
 | |
| }
 | |
| 
 | |
| func (repos RepositoryList) Less(i, j int) bool {
 | |
| 	return repos[i].FullName() < repos[j].FullName()
 | |
| }
 | |
| 
 | |
| func (repos RepositoryList) Swap(i, j int) {
 | |
| 	repos[i], repos[j] = repos[j], repos[i]
 | |
| }
 | |
| 
 | |
| // RepositoryListOfMap make list from values of map
 | |
| func RepositoryListOfMap(repoMap map[int64]*Repository) RepositoryList {
 | |
| 	return RepositoryList(valuesRepository(repoMap))
 | |
| }
 | |
| 
 | |
| func (repos RepositoryList) loadAttributes(e Engine) error {
 | |
| 	if len(repos) == 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	// Load owners.
 | |
| 	set := make(map[int64]struct{})
 | |
| 	for i := range repos {
 | |
| 		set[repos[i].OwnerID] = struct{}{}
 | |
| 	}
 | |
| 	users := make(map[int64]*User, len(set))
 | |
| 	if err := e.
 | |
| 		Where("id > 0").
 | |
| 		In("id", keysInt64(set)).
 | |
| 		Find(&users); err != nil {
 | |
| 		return fmt.Errorf("find users: %v", err)
 | |
| 	}
 | |
| 	for i := range repos {
 | |
| 		repos[i].Owner = users[repos[i].OwnerID]
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // LoadAttributes loads the attributes for the given RepositoryList
 | |
| func (repos RepositoryList) LoadAttributes() error {
 | |
| 	return repos.loadAttributes(x)
 | |
| }
 | |
| 
 | |
| // MirrorRepositoryList contains the mirror repositories
 | |
| type MirrorRepositoryList []*Repository
 | |
| 
 | |
| func (repos MirrorRepositoryList) loadAttributes(e Engine) error {
 | |
| 	if len(repos) == 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	// Load mirrors.
 | |
| 	repoIDs := make([]int64, 0, len(repos))
 | |
| 	for i := range repos {
 | |
| 		if !repos[i].IsMirror {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		repoIDs = append(repoIDs, repos[i].ID)
 | |
| 	}
 | |
| 	mirrors := make([]*Mirror, 0, len(repoIDs))
 | |
| 	if err := e.
 | |
| 		Where("id > 0").
 | |
| 		In("repo_id", repoIDs).
 | |
| 		Find(&mirrors); err != nil {
 | |
| 		return fmt.Errorf("find mirrors: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	set := make(map[int64]*Mirror)
 | |
| 	for i := range mirrors {
 | |
| 		set[mirrors[i].RepoID] = mirrors[i]
 | |
| 	}
 | |
| 	for i := range repos {
 | |
| 		repos[i].Mirror = set[repos[i].ID]
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // LoadAttributes loads the attributes for the given MirrorRepositoryList
 | |
| func (repos MirrorRepositoryList) LoadAttributes() error {
 | |
| 	return repos.loadAttributes(x)
 | |
| }
 | |
| 
 | |
| // SearchRepoOptions holds the search options
 | |
| type SearchRepoOptions struct {
 | |
| 	ListOptions
 | |
| 	Actor           *User
 | |
| 	Keyword         string
 | |
| 	OwnerID         int64
 | |
| 	PriorityOwnerID int64
 | |
| 	OrderBy         SearchOrderBy
 | |
| 	Private         bool // Include private repositories in results
 | |
| 	StarredByID     int64
 | |
| 	IsProfile       bool
 | |
| 	AllPublic       bool // Include also all public repositories of users and public organisations
 | |
| 	AllLimited      bool // Include also all public repositories of limited organisations
 | |
| 	// None -> include collaborative AND non-collaborative
 | |
| 	// True -> include just collaborative
 | |
| 	// False -> incude just non-collaborative
 | |
| 	Collaborate util.OptionalBool
 | |
| 	// None -> include forks AND non-forks
 | |
| 	// True -> include just forks
 | |
| 	// False -> include just non-forks
 | |
| 	Fork util.OptionalBool
 | |
| 	// None -> include templates AND non-templates
 | |
| 	// True -> include just templates
 | |
| 	// False -> include just non-templates
 | |
| 	Template util.OptionalBool
 | |
| 	// None -> include mirrors AND non-mirrors
 | |
| 	// True -> include just mirrors
 | |
| 	// False -> include just non-mirrors
 | |
| 	Mirror util.OptionalBool
 | |
| 	// only search topic name
 | |
| 	TopicOnly bool
 | |
| 	// include description in keyword search
 | |
| 	IncludeDescription bool
 | |
| }
 | |
| 
 | |
| //SearchOrderBy is used to sort the result
 | |
| type SearchOrderBy string
 | |
| 
 | |
| func (s SearchOrderBy) String() string {
 | |
| 	return string(s)
 | |
| }
 | |
| 
 | |
| // Strings for sorting result
 | |
| const (
 | |
| 	SearchOrderByAlphabetically        SearchOrderBy = "name ASC"
 | |
| 	SearchOrderByAlphabeticallyReverse SearchOrderBy = "name DESC"
 | |
| 	SearchOrderByLeastUpdated          SearchOrderBy = "updated_unix ASC"
 | |
| 	SearchOrderByRecentUpdated         SearchOrderBy = "updated_unix DESC"
 | |
| 	SearchOrderByOldest                SearchOrderBy = "created_unix ASC"
 | |
| 	SearchOrderByNewest                SearchOrderBy = "created_unix DESC"
 | |
| 	SearchOrderBySize                  SearchOrderBy = "size ASC"
 | |
| 	SearchOrderBySizeReverse           SearchOrderBy = "size DESC"
 | |
| 	SearchOrderByID                    SearchOrderBy = "id ASC"
 | |
| 	SearchOrderByIDReverse             SearchOrderBy = "id DESC"
 | |
| 	SearchOrderByStars                 SearchOrderBy = "num_stars ASC"
 | |
| 	SearchOrderByStarsReverse          SearchOrderBy = "num_stars DESC"
 | |
| 	SearchOrderByForks                 SearchOrderBy = "num_forks ASC"
 | |
| 	SearchOrderByForksReverse          SearchOrderBy = "num_forks DESC"
 | |
| )
 | |
| 
 | |
| // SearchRepository returns repositories based on search options,
 | |
| // it returns results in given range and number of total results.
 | |
| func SearchRepository(opts *SearchRepoOptions) (RepositoryList, int64, error) {
 | |
| 	if opts.Page <= 0 {
 | |
| 		opts.Page = 1
 | |
| 	}
 | |
| 	var cond = builder.NewCond()
 | |
| 
 | |
| 	if opts.Private {
 | |
| 		if opts.Actor != nil && !opts.Actor.IsAdmin && opts.Actor.ID != opts.OwnerID {
 | |
| 			// OK we're in the context of a User
 | |
| 			cond = cond.And(accessibleRepositoryCondition(opts.Actor))
 | |
| 		}
 | |
| 	} else {
 | |
| 		// Not looking at private organisations
 | |
| 		// We should be able to see all non-private repositories that either:
 | |
| 		cond = cond.And(builder.Eq{"is_private": false})
 | |
| 		accessCond := builder.Or(
 | |
| 			//   A. Aren't in organisations  __OR__
 | |
| 			builder.NotIn("owner_id", builder.Select("id").From("`user`").Where(builder.Eq{"type": UserTypeOrganization})),
 | |
| 			//   B. Isn't a private or limited organisation.
 | |
| 			builder.NotIn("owner_id", builder.Select("id").From("`user`").Where(builder.Or(builder.Eq{"visibility": structs.VisibleTypeLimited}, builder.Eq{"visibility": structs.VisibleTypePrivate}))))
 | |
| 		cond = cond.And(accessCond)
 | |
| 	}
 | |
| 
 | |
| 	if opts.Template != util.OptionalBoolNone {
 | |
| 		cond = cond.And(builder.Eq{"is_template": opts.Template == util.OptionalBoolTrue})
 | |
| 	}
 | |
| 
 | |
| 	// Restrict to starred repositories
 | |
| 	if opts.StarredByID > 0 {
 | |
| 		cond = cond.And(builder.In("id", builder.Select("repo_id").From("star").Where(builder.Eq{"uid": opts.StarredByID})))
 | |
| 	}
 | |
| 
 | |
| 	// Restrict repositories to those the OwnerID owns or contributes to as per opts.Collaborate
 | |
| 	if opts.OwnerID > 0 {
 | |
| 		var accessCond = builder.NewCond()
 | |
| 		if opts.Collaborate != util.OptionalBoolTrue {
 | |
| 			accessCond = builder.Eq{"owner_id": opts.OwnerID}
 | |
| 		}
 | |
| 
 | |
| 		if opts.Collaborate != util.OptionalBoolFalse {
 | |
| 			collaborateCond := builder.And(
 | |
| 				builder.Or(
 | |
| 					builder.Expr("repository.id IN (SELECT repo_id FROM `access` WHERE access.user_id = ?)", opts.OwnerID),
 | |
| 					builder.In("id", builder.Select("`team_repo`.repo_id").
 | |
| 						From("team_repo").
 | |
| 						Where(builder.Eq{"`team_user`.uid": opts.OwnerID}).
 | |
| 						Join("INNER", "team_user", "`team_user`.team_id = `team_repo`.team_id"))),
 | |
| 				builder.Neq{"owner_id": opts.OwnerID})
 | |
| 			if !opts.Private {
 | |
| 				collaborateCond = collaborateCond.And(builder.Expr("owner_id NOT IN (SELECT org_id FROM org_user WHERE org_user.uid = ? AND org_user.is_public = ?)", opts.OwnerID, false))
 | |
| 			}
 | |
| 
 | |
| 			accessCond = accessCond.Or(collaborateCond)
 | |
| 		}
 | |
| 
 | |
| 		if opts.AllPublic {
 | |
| 			accessCond = accessCond.Or(builder.Eq{"is_private": false}.And(builder.In("owner_id", builder.Select("`user`.id").From("`user`").Where(builder.Eq{"`user`.visibility": structs.VisibleTypePublic}))))
 | |
| 		}
 | |
| 
 | |
| 		if opts.AllLimited {
 | |
| 			accessCond = accessCond.Or(builder.Eq{"is_private": false}.And(builder.In("owner_id", builder.Select("`user`.id").From("`user`").Where(builder.Eq{"`user`.visibility": structs.VisibleTypeLimited}))))
 | |
| 		}
 | |
| 
 | |
| 		cond = cond.And(accessCond)
 | |
| 	}
 | |
| 
 | |
| 	if opts.Keyword != "" {
 | |
| 		// separate keyword
 | |
| 		var subQueryCond = builder.NewCond()
 | |
| 		for _, v := range strings.Split(opts.Keyword, ",") {
 | |
| 			if opts.TopicOnly {
 | |
| 				subQueryCond = subQueryCond.Or(builder.Eq{"topic.name": strings.ToLower(v)})
 | |
| 			} else {
 | |
| 				subQueryCond = subQueryCond.Or(builder.Like{"topic.name", strings.ToLower(v)})
 | |
| 			}
 | |
| 		}
 | |
| 		subQuery := builder.Select("repo_topic.repo_id").From("repo_topic").
 | |
| 			Join("INNER", "topic", "topic.id = repo_topic.topic_id").
 | |
| 			Where(subQueryCond).
 | |
| 			GroupBy("repo_topic.repo_id")
 | |
| 
 | |
| 		var keywordCond = builder.In("id", subQuery)
 | |
| 		if !opts.TopicOnly {
 | |
| 			var likes = builder.NewCond()
 | |
| 			for _, v := range strings.Split(opts.Keyword, ",") {
 | |
| 				likes = likes.Or(builder.Like{"lower_name", strings.ToLower(v)})
 | |
| 				if opts.IncludeDescription {
 | |
| 					likes = likes.Or(builder.Like{"LOWER(description)", strings.ToLower(v)})
 | |
| 				}
 | |
| 			}
 | |
| 			keywordCond = keywordCond.Or(likes)
 | |
| 		}
 | |
| 		cond = cond.And(keywordCond)
 | |
| 	}
 | |
| 
 | |
| 	if opts.Fork != util.OptionalBoolNone {
 | |
| 		cond = cond.And(builder.Eq{"is_fork": opts.Fork == util.OptionalBoolTrue})
 | |
| 	}
 | |
| 
 | |
| 	if opts.Mirror != util.OptionalBoolNone {
 | |
| 		cond = cond.And(builder.Eq{"is_mirror": opts.Mirror == util.OptionalBoolTrue})
 | |
| 	}
 | |
| 
 | |
| 	if opts.Actor != nil && opts.Actor.IsRestricted {
 | |
| 		cond = cond.And(accessibleRepositoryCondition(opts.Actor))
 | |
| 	}
 | |
| 
 | |
| 	if len(opts.OrderBy) == 0 {
 | |
| 		opts.OrderBy = SearchOrderByAlphabetically
 | |
| 	}
 | |
| 
 | |
| 	if opts.PriorityOwnerID > 0 {
 | |
| 		opts.OrderBy = SearchOrderBy(fmt.Sprintf("CASE WHEN owner_id = %d THEN 0 ELSE owner_id END, %s", opts.PriorityOwnerID, opts.OrderBy))
 | |
| 	}
 | |
| 
 | |
| 	sess := x.NewSession()
 | |
| 	defer sess.Close()
 | |
| 
 | |
| 	count, err := sess.
 | |
| 		Where(cond).
 | |
| 		Count(new(Repository))
 | |
| 
 | |
| 	if err != nil {
 | |
| 		return nil, 0, fmt.Errorf("Count: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	repos := make(RepositoryList, 0, opts.PageSize)
 | |
| 	if err = sess.
 | |
| 		Where(cond).
 | |
| 		OrderBy(opts.OrderBy.String()).
 | |
| 		Limit(opts.PageSize, (opts.Page-1)*opts.PageSize).
 | |
| 		Find(&repos); err != nil {
 | |
| 		return nil, 0, fmt.Errorf("Repo: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	if !opts.IsProfile {
 | |
| 		if err = repos.loadAttributes(sess); err != nil {
 | |
| 			return nil, 0, fmt.Errorf("LoadAttributes: %v", err)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return repos, count, nil
 | |
| }
 | |
| 
 | |
| // accessibleRepositoryCondition takes a user a returns a condition for checking if a repository is accessible
 | |
| func accessibleRepositoryCondition(user *User) builder.Cond {
 | |
| 	var cond = builder.NewCond()
 | |
| 
 | |
| 	if user == nil || !user.IsRestricted {
 | |
| 		orgVisibilityLimit := []structs.VisibleType{structs.VisibleTypePrivate}
 | |
| 		if user == nil {
 | |
| 			orgVisibilityLimit = append(orgVisibilityLimit, structs.VisibleTypeLimited)
 | |
| 		}
 | |
| 		// 1. Be able to see all non-private repositories that either:
 | |
| 		cond = cond.Or(builder.And(
 | |
| 			builder.Eq{"`repository`.is_private": false},
 | |
| 			builder.Or(
 | |
| 				//   A. Aren't in organisations  __OR__
 | |
| 				builder.NotIn("`repository`.owner_id", builder.Select("id").From("`user`").Where(builder.Eq{"type": UserTypeOrganization})),
 | |
| 				//   B. Isn't a private organisation. Limited is OK as long as we're logged in.
 | |
| 				builder.NotIn("`repository`.owner_id", builder.Select("id").From("`user`").Where(builder.In("visibility", orgVisibilityLimit))))))
 | |
| 	}
 | |
| 
 | |
| 	if user != nil {
 | |
| 		// 2. Be able to see all repositories that we have access to
 | |
| 		cond = cond.Or(builder.Or(
 | |
| 			builder.In("`repository`.id", builder.Select("repo_id").
 | |
| 				From("`access`").
 | |
| 				Where(builder.And(
 | |
| 					builder.Eq{"user_id": user.ID},
 | |
| 					builder.Gt{"mode": int(AccessModeNone)}))),
 | |
| 			builder.In("`repository`.id", builder.Select("id").
 | |
| 				From("`repository`").
 | |
| 				Where(builder.Eq{"owner_id": user.ID}))))
 | |
| 		// 3. Be able to see all repositories that we are in a team
 | |
| 		cond = cond.Or(builder.In("`repository`.id", builder.Select("`team_repo`.repo_id").
 | |
| 			From("team_repo").
 | |
| 			Where(builder.Eq{"`team_user`.uid": user.ID}).
 | |
| 			Join("INNER", "team_user", "`team_user`.team_id = `team_repo`.team_id")))
 | |
| 	}
 | |
| 
 | |
| 	return cond
 | |
| }
 | |
| 
 | |
| // SearchRepositoryByName takes keyword and part of repository name to search,
 | |
| // it returns results in given range and number of total results.
 | |
| func SearchRepositoryByName(opts *SearchRepoOptions) (RepositoryList, int64, error) {
 | |
| 	opts.IncludeDescription = false
 | |
| 	return SearchRepository(opts)
 | |
| }
 | |
| 
 | |
| // AccessibleRepoIDsQuery queries accessible repository ids. Usable as a subquery wherever repo ids need to be filtered.
 | |
| func (user *User) AccessibleRepoIDsQuery() *builder.Builder {
 | |
| 	return builder.Select("id").From("repository").Where(accessibleRepositoryCondition(user))
 | |
| }
 | |
| 
 | |
| // FindUserAccessibleRepoIDs find all accessible repositories' ID by user's id
 | |
| func FindUserAccessibleRepoIDs(user *User) ([]int64, error) {
 | |
| 	repoIDs := make([]int64, 0, 10)
 | |
| 	if err := x.
 | |
| 		Table("repository").
 | |
| 		Cols("id").
 | |
| 		Where(accessibleRepositoryCondition(user)).
 | |
| 		Find(&repoIDs); err != nil {
 | |
| 		return nil, fmt.Errorf("FindUserAccesibleRepoIDs: %v", err)
 | |
| 	}
 | |
| 	return repoIDs, nil
 | |
| }
 |