Refactor for issues loadattributes of a repository (#971)
* refactor for issues loadattributes of a repository * refactors
This commit is contained in:
parent
29c6f32a3b
commit
1f7837d6d6
3 changed files with 343 additions and 64 deletions
21
models/helper.go
Normal file
21
models/helper.go
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// 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
|
||||||
|
|
||||||
|
func keysInt64(m map[int64]struct{}) []int64 {
|
||||||
|
var keys = make([]int64, 0, len(m))
|
||||||
|
for k, _ := range m {
|
||||||
|
keys = append(keys, k)
|
||||||
|
}
|
||||||
|
return keys
|
||||||
|
}
|
||||||
|
|
||||||
|
func valuesRepository(m map[int64]*Repository) []*Repository {
|
||||||
|
var values = make([]*Repository, 0, len(m))
|
||||||
|
for _, v := range m {
|
||||||
|
values = append(values, v)
|
||||||
|
}
|
||||||
|
return values
|
||||||
|
}
|
|
@ -1128,11 +1128,8 @@ func Issues(opts *IssuesOptions) ([]*Issue, error) {
|
||||||
return nil, fmt.Errorf("Find: %v", err)
|
return nil, fmt.Errorf("Find: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: use IssueList to improve performance.
|
if err := IssueList(issues).LoadAttributes(); err != nil {
|
||||||
for i := range issues {
|
return nil, fmt.Errorf("LoadAttributes: %v", err)
|
||||||
if err := issues[i].LoadAttributes(); err != nil {
|
|
||||||
return nil, fmt.Errorf("LoadAttributes [%d]: %v", issues[i].ID, err)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return issues, nil
|
return issues, nil
|
||||||
|
@ -1399,62 +1396,3 @@ func updateIssue(e Engine, issue *Issue) error {
|
||||||
func UpdateIssue(issue *Issue) error {
|
func UpdateIssue(issue *Issue) error {
|
||||||
return updateIssue(x, issue)
|
return updateIssue(x, issue)
|
||||||
}
|
}
|
||||||
|
|
||||||
// IssueList defines a list of issues
|
|
||||||
type IssueList []*Issue
|
|
||||||
|
|
||||||
func (issues IssueList) getRepoIDs() []int64 {
|
|
||||||
repoIDs := make([]int64, 0, len(issues))
|
|
||||||
for _, issue := range issues {
|
|
||||||
var has bool
|
|
||||||
for _, repoID := range repoIDs {
|
|
||||||
if repoID == issue.RepoID {
|
|
||||||
has = true
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !has {
|
|
||||||
repoIDs = append(repoIDs, issue.RepoID)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return repoIDs
|
|
||||||
}
|
|
||||||
|
|
||||||
func (issues IssueList) loadRepositories(e Engine) ([]*Repository, error) {
|
|
||||||
if len(issues) == 0 {
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
repoIDs := issues.getRepoIDs()
|
|
||||||
rows, err := e.
|
|
||||||
Where("id > 0").
|
|
||||||
In("id", repoIDs).
|
|
||||||
Rows(new(Repository))
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("find repository: %v", err)
|
|
||||||
}
|
|
||||||
defer rows.Close()
|
|
||||||
|
|
||||||
repositories := make([]*Repository, 0, len(repoIDs))
|
|
||||||
repoMaps := make(map[int64]*Repository, len(repoIDs))
|
|
||||||
for rows.Next() {
|
|
||||||
var repo Repository
|
|
||||||
err = rows.Scan(&repo)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("find repository: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
repositories = append(repositories, &repo)
|
|
||||||
repoMaps[repo.ID] = &repo
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, issue := range issues {
|
|
||||||
issue.Repo = repoMaps[issue.RepoID]
|
|
||||||
}
|
|
||||||
return repositories, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// LoadRepositories loads issues' all repositories
|
|
||||||
func (issues IssueList) LoadRepositories() ([]*Repository, error) {
|
|
||||||
return issues.loadRepositories(x)
|
|
||||||
}
|
|
||||||
|
|
320
models/issue_list.go
Normal file
320
models/issue_list.go
Normal file
|
@ -0,0 +1,320 @@
|
||||||
|
// 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"
|
||||||
|
|
||||||
|
// IssueList defines a list of issues
|
||||||
|
type IssueList []*Issue
|
||||||
|
|
||||||
|
func (issues IssueList) getRepoIDs() []int64 {
|
||||||
|
repoIDs := make(map[int64]struct{}, len(issues))
|
||||||
|
for _, issue := range issues {
|
||||||
|
if _, ok := repoIDs[issue.RepoID]; !ok {
|
||||||
|
repoIDs[issue.RepoID] = struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return keysInt64(repoIDs)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) loadRepositories(e Engine) ([]*Repository, error) {
|
||||||
|
if len(issues) == 0 {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
repoIDs := issues.getRepoIDs()
|
||||||
|
repoMaps := make(map[int64]*Repository, len(repoIDs))
|
||||||
|
err := e.
|
||||||
|
In("id", repoIDs).
|
||||||
|
Find(&repoMaps)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("find repository: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, issue := range issues {
|
||||||
|
issue.Repo = repoMaps[issue.RepoID]
|
||||||
|
}
|
||||||
|
return valuesRepository(repoMaps), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// LoadRepositories loads issues' all repositories
|
||||||
|
func (issues IssueList) LoadRepositories() ([]*Repository, error) {
|
||||||
|
return issues.loadRepositories(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) getPosterIDs() []int64 {
|
||||||
|
posterIDs := make(map[int64]struct{}, len(issues))
|
||||||
|
for _, issue := range issues {
|
||||||
|
if _, ok := posterIDs[issue.PosterID]; !ok {
|
||||||
|
posterIDs[issue.PosterID] = struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return keysInt64(posterIDs)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) loadPosters(e Engine) error {
|
||||||
|
if len(issues) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
postgerIDs := issues.getPosterIDs()
|
||||||
|
posterMaps := make(map[int64]*User, len(postgerIDs))
|
||||||
|
err := e.
|
||||||
|
In("id", postgerIDs).
|
||||||
|
Find(&posterMaps)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, issue := range issues {
|
||||||
|
issue.Poster = posterMaps[issue.PosterID]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) getIssueIDs() []int64 {
|
||||||
|
var ids = make([]int64, 0, len(issues))
|
||||||
|
for _, issue := range issues {
|
||||||
|
ids = append(ids, issue.ID)
|
||||||
|
}
|
||||||
|
return ids
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) loadLabels(e Engine) error {
|
||||||
|
if len(issues) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type LabelIssue struct {
|
||||||
|
Label *Label `xorm:"extends"`
|
||||||
|
IssueLabel *IssueLabel `xorm:"extends"`
|
||||||
|
}
|
||||||
|
|
||||||
|
var issueLabels = make(map[int64][]*Label, len(issues)*3)
|
||||||
|
rows, err := e.Table("label").
|
||||||
|
Join("LEFT", "issue_label", "issue_label.label_id = label.id").
|
||||||
|
In("issue_label.issue_id", issues.getIssueIDs()).
|
||||||
|
Asc("label.name").
|
||||||
|
Rows(new(LabelIssue))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer rows.Close()
|
||||||
|
|
||||||
|
for rows.Next() {
|
||||||
|
var labelIssue LabelIssue
|
||||||
|
err = rows.Scan(&labelIssue)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
issueLabels[labelIssue.IssueLabel.IssueID] = append(issueLabels[labelIssue.IssueLabel.IssueID], labelIssue.Label)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, issue := range issues {
|
||||||
|
issue.Labels = issueLabels[issue.ID]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) getMilestoneIDs() []int64 {
|
||||||
|
var ids = make(map[int64]struct{}, len(issues))
|
||||||
|
for _, issue := range issues {
|
||||||
|
if _, ok := ids[issue.MilestoneID]; !ok {
|
||||||
|
ids[issue.MilestoneID] = struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return keysInt64(ids)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) loadMilestones(e Engine) error {
|
||||||
|
milestoneIDs := issues.getMilestoneIDs()
|
||||||
|
if len(milestoneIDs) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
milestoneMaps := make(map[int64]*Milestone, len(milestoneIDs))
|
||||||
|
err := e.
|
||||||
|
In("id", milestoneIDs).
|
||||||
|
Find(&milestoneMaps)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, issue := range issues {
|
||||||
|
issue.Milestone = milestoneMaps[issue.MilestoneID]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) getAssigneeIDs() []int64 {
|
||||||
|
var ids = make(map[int64]struct{}, len(issues))
|
||||||
|
for _, issue := range issues {
|
||||||
|
if _, ok := ids[issue.AssigneeID]; !ok {
|
||||||
|
ids[issue.AssigneeID] = struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return keysInt64(ids)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) loadAssignees(e Engine) error {
|
||||||
|
assigneeIDs := issues.getAssigneeIDs()
|
||||||
|
if len(assigneeIDs) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
assigneeMaps := make(map[int64]*User, len(assigneeIDs))
|
||||||
|
err := e.
|
||||||
|
In("id", assigneeIDs).
|
||||||
|
Find(&assigneeMaps)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, issue := range issues {
|
||||||
|
issue.Assignee = assigneeMaps[issue.AssigneeID]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) getPullIssueIDs() []int64 {
|
||||||
|
var ids = make([]int64, 0, len(issues))
|
||||||
|
for _, issue := range issues {
|
||||||
|
if issue.IsPull && issue.PullRequest == nil {
|
||||||
|
ids = append(ids, issue.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ids
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) loadPullRequests(e Engine) error {
|
||||||
|
issuesIDs := issues.getPullIssueIDs()
|
||||||
|
if len(issuesIDs) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
pullRequestMaps := make(map[int64]*PullRequest, len(issuesIDs))
|
||||||
|
rows, err := e.
|
||||||
|
In("issue_id", issuesIDs).
|
||||||
|
Rows(new(PullRequest))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer rows.Close()
|
||||||
|
|
||||||
|
for rows.Next() {
|
||||||
|
var pr PullRequest
|
||||||
|
err = rows.Scan(&pr)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
pullRequestMaps[pr.IssueID] = &pr
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, issue := range issues {
|
||||||
|
issue.PullRequest = pullRequestMaps[issue.ID]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) loadAttachments(e Engine) (err error) {
|
||||||
|
if len(issues) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var attachments = make(map[int64][]*Attachment, len(issues))
|
||||||
|
rows, err := e.Table("attachment").
|
||||||
|
Join("INNER", "issue", "issue.id = attachment.issue_id").
|
||||||
|
In("issue.id", issues.getIssueIDs()).
|
||||||
|
Rows(new(Attachment))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer rows.Close()
|
||||||
|
|
||||||
|
for rows.Next() {
|
||||||
|
var attachment Attachment
|
||||||
|
err = rows.Scan(&attachment)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
attachments[attachment.IssueID] = append(attachments[attachment.IssueID], &attachment)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, issue := range issues {
|
||||||
|
issue.Attachments = attachments[issue.ID]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) loadComments(e Engine) (err error) {
|
||||||
|
if len(issues) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var comments = make(map[int64][]*Comment, len(issues))
|
||||||
|
rows, err := e.Table("comment").
|
||||||
|
Join("INNER", "issue", "issue.id = comment.issue_id").
|
||||||
|
In("issue.id", issues.getIssueIDs()).
|
||||||
|
Rows(new(Comment))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer rows.Close()
|
||||||
|
|
||||||
|
for rows.Next() {
|
||||||
|
var comment Comment
|
||||||
|
err = rows.Scan(&comment)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
comments[comment.IssueID] = append(comments[comment.IssueID], &comment)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, issue := range issues {
|
||||||
|
issue.Comments = comments[issue.ID]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (issues IssueList) loadAttributes(e Engine) (err error) {
|
||||||
|
if _, err = issues.loadRepositories(e); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = issues.loadPosters(e); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = issues.loadLabels(e); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = issues.loadMilestones(e); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = issues.loadAssignees(e); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = issues.loadPullRequests(e); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = issues.loadAttachments(e); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = issues.loadComments(e); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// LoadAttributes loads atrributes of the issues
|
||||||
|
func (issues IssueList) LoadAttributes() error {
|
||||||
|
return issues.loadAttributes(x)
|
||||||
|
}
|
Reference in a new issue