2020-01-11 10:59:41 +08:00
|
|
|
// Copyright 2020 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 convert
|
|
|
|
|
|
|
|
import (
|
2021-09-10 18:03:16 +02:00
|
|
|
"fmt"
|
2021-11-16 18:18:25 +00:00
|
|
|
"net/url"
|
2020-02-29 03:49:50 +01:00
|
|
|
"strings"
|
|
|
|
|
2021-12-10 09:27:50 +08:00
|
|
|
"code.gitea.io/gitea/models/db"
|
2022-04-08 17:11:15 +08:00
|
|
|
issues_model "code.gitea.io/gitea/models/issues"
|
2021-12-10 09:27:50 +08:00
|
|
|
repo_model "code.gitea.io/gitea/models/repo"
|
2021-11-24 17:49:20 +08:00
|
|
|
user_model "code.gitea.io/gitea/models/user"
|
2021-09-10 18:03:16 +02:00
|
|
|
"code.gitea.io/gitea/modules/log"
|
|
|
|
"code.gitea.io/gitea/modules/setting"
|
2020-01-11 10:59:41 +08:00
|
|
|
api "code.gitea.io/gitea/modules/structs"
|
|
|
|
)
|
|
|
|
|
2020-02-29 03:49:50 +01:00
|
|
|
// ToAPIIssue converts an Issue to API format
|
|
|
|
// it assumes some fields assigned with values:
|
|
|
|
// Required - Poster, Labels,
|
|
|
|
// Optional - Milestone, Assignee, PullRequest
|
2022-06-13 17:37:59 +08:00
|
|
|
func ToAPIIssue(issue *issues_model.Issue) *api.Issue {
|
2022-04-28 13:48:48 +02:00
|
|
|
if err := issue.LoadLabels(db.DefaultContext); err != nil {
|
2020-02-29 03:49:50 +01:00
|
|
|
return &api.Issue{}
|
|
|
|
}
|
|
|
|
if err := issue.LoadPoster(); err != nil {
|
|
|
|
return &api.Issue{}
|
|
|
|
}
|
2022-04-08 17:11:15 +08:00
|
|
|
if err := issue.LoadRepo(db.DefaultContext); err != nil {
|
2020-02-29 03:49:50 +01:00
|
|
|
return &api.Issue{}
|
|
|
|
}
|
2021-12-10 09:27:50 +08:00
|
|
|
if err := issue.Repo.GetOwner(db.DefaultContext); err != nil {
|
2021-09-10 18:03:16 +02:00
|
|
|
return &api.Issue{}
|
|
|
|
}
|
2020-02-29 03:49:50 +01:00
|
|
|
|
|
|
|
apiIssue := &api.Issue{
|
|
|
|
ID: issue.ID,
|
|
|
|
URL: issue.APIURL(),
|
|
|
|
HTMLURL: issue.HTMLURL(),
|
|
|
|
Index: issue.Index,
|
2021-03-27 17:45:26 +01:00
|
|
|
Poster: ToUser(issue.Poster, nil),
|
2020-02-29 03:49:50 +01:00
|
|
|
Title: issue.Title,
|
|
|
|
Body: issue.Content,
|
2020-12-13 11:34:11 +00:00
|
|
|
Ref: issue.Ref,
|
2021-09-10 18:03:16 +02:00
|
|
|
Labels: ToLabelList(issue.Labels, issue.Repo, issue.Repo.Owner),
|
2020-02-29 03:49:50 +01:00
|
|
|
State: issue.State(),
|
2020-06-01 23:01:55 +02:00
|
|
|
IsLocked: issue.IsLocked,
|
2020-02-29 03:49:50 +01:00
|
|
|
Comments: issue.NumComments,
|
|
|
|
Created: issue.CreatedUnix.AsTime(),
|
|
|
|
Updated: issue.UpdatedUnix.AsTime(),
|
|
|
|
}
|
|
|
|
|
|
|
|
apiIssue.Repo = &api.RepositoryMeta{
|
|
|
|
ID: issue.Repo.ID,
|
|
|
|
Name: issue.Repo.Name,
|
|
|
|
Owner: issue.Repo.OwnerName,
|
|
|
|
FullName: issue.Repo.FullName(),
|
|
|
|
}
|
|
|
|
|
|
|
|
if issue.ClosedUnix != 0 {
|
|
|
|
apiIssue.Closed = issue.ClosedUnix.AsTimePtr()
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := issue.LoadMilestone(); err != nil {
|
|
|
|
return &api.Issue{}
|
|
|
|
}
|
|
|
|
if issue.Milestone != nil {
|
2020-05-12 23:54:35 +02:00
|
|
|
apiIssue.Milestone = ToAPIMilestone(issue.Milestone)
|
2020-02-29 03:49:50 +01:00
|
|
|
}
|
|
|
|
|
2022-05-20 22:08:52 +08:00
|
|
|
if err := issue.LoadAssignees(db.DefaultContext); err != nil {
|
2020-02-29 03:49:50 +01:00
|
|
|
return &api.Issue{}
|
|
|
|
}
|
|
|
|
if len(issue.Assignees) > 0 {
|
|
|
|
for _, assignee := range issue.Assignees {
|
2021-03-27 17:45:26 +01:00
|
|
|
apiIssue.Assignees = append(apiIssue.Assignees, ToUser(assignee, nil))
|
2020-02-29 03:49:50 +01:00
|
|
|
}
|
2021-03-27 17:45:26 +01:00
|
|
|
apiIssue.Assignee = ToUser(issue.Assignees[0], nil) // For compatibility, we're keeping the first assignee as `apiIssue.Assignee`
|
2020-02-29 03:49:50 +01:00
|
|
|
}
|
|
|
|
if issue.IsPull {
|
|
|
|
if err := issue.LoadPullRequest(); err != nil {
|
|
|
|
return &api.Issue{}
|
|
|
|
}
|
|
|
|
apiIssue.PullRequest = &api.PullRequestMeta{
|
|
|
|
HasMerged: issue.PullRequest.HasMerged,
|
|
|
|
}
|
|
|
|
if issue.PullRequest.HasMerged {
|
|
|
|
apiIssue.PullRequest.Merged = issue.PullRequest.MergedUnix.AsTimePtr()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if issue.DeadlineUnix != 0 {
|
|
|
|
apiIssue.Deadline = issue.DeadlineUnix.AsTimePtr()
|
|
|
|
}
|
|
|
|
|
|
|
|
return apiIssue
|
|
|
|
}
|
|
|
|
|
|
|
|
// ToAPIIssueList converts an IssueList to API format
|
2022-06-13 17:37:59 +08:00
|
|
|
func ToAPIIssueList(il issues_model.IssueList) []*api.Issue {
|
2020-02-29 03:49:50 +01:00
|
|
|
result := make([]*api.Issue, len(il))
|
|
|
|
for i := range il {
|
|
|
|
result[i] = ToAPIIssue(il[i])
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2020-01-11 10:59:41 +08:00
|
|
|
// ToTrackedTime converts TrackedTime to API format
|
2022-06-13 17:37:59 +08:00
|
|
|
func ToTrackedTime(t *issues_model.TrackedTime) (apiT *api.TrackedTime) {
|
2020-01-11 10:59:41 +08:00
|
|
|
apiT = &api.TrackedTime{
|
|
|
|
ID: t.ID,
|
|
|
|
IssueID: t.IssueID,
|
|
|
|
UserID: t.UserID,
|
|
|
|
UserName: t.User.Name,
|
|
|
|
Time: t.Time,
|
|
|
|
Created: t.Created,
|
|
|
|
}
|
|
|
|
if t.Issue != nil {
|
2020-02-29 03:49:50 +01:00
|
|
|
apiT.Issue = ToAPIIssue(t.Issue)
|
2020-01-11 10:59:41 +08:00
|
|
|
}
|
|
|
|
if t.User != nil {
|
|
|
|
apiT.UserName = t.User.Name
|
|
|
|
}
|
2022-06-20 12:02:49 +02:00
|
|
|
return apiT
|
2020-01-11 10:59:41 +08:00
|
|
|
}
|
|
|
|
|
2020-09-18 14:09:26 +02:00
|
|
|
// ToStopWatches convert Stopwatch list to api.StopWatches
|
2022-06-13 17:37:59 +08:00
|
|
|
func ToStopWatches(sws []*issues_model.Stopwatch) (api.StopWatches, error) {
|
2020-09-18 14:09:26 +02:00
|
|
|
result := api.StopWatches(make([]api.StopWatch, 0, len(sws)))
|
|
|
|
|
2022-06-13 17:37:59 +08:00
|
|
|
issueCache := make(map[int64]*issues_model.Issue)
|
2021-12-10 09:27:50 +08:00
|
|
|
repoCache := make(map[int64]*repo_model.Repository)
|
2020-09-18 14:09:26 +02:00
|
|
|
var (
|
2022-06-13 17:37:59 +08:00
|
|
|
issue *issues_model.Issue
|
2021-12-10 09:27:50 +08:00
|
|
|
repo *repo_model.Repository
|
2020-09-18 14:09:26 +02:00
|
|
|
ok bool
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
for _, sw := range sws {
|
|
|
|
issue, ok = issueCache[sw.IssueID]
|
|
|
|
if !ok {
|
2022-06-13 17:37:59 +08:00
|
|
|
issue, err = issues_model.GetIssueByID(db.DefaultContext, sw.IssueID)
|
2020-09-18 14:09:26 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
repo, ok = repoCache[issue.RepoID]
|
|
|
|
if !ok {
|
2021-12-10 09:27:50 +08:00
|
|
|
repo, err = repo_model.GetRepositoryByID(issue.RepoID)
|
2020-09-18 14:09:26 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
result = append(result, api.StopWatch{
|
|
|
|
Created: sw.CreatedUnix.AsTime(),
|
2021-01-21 14:51:52 +00:00
|
|
|
Seconds: sw.Seconds(),
|
|
|
|
Duration: sw.Duration(),
|
2020-09-18 14:09:26 +02:00
|
|
|
IssueIndex: issue.Index,
|
|
|
|
IssueTitle: issue.Title,
|
|
|
|
RepoOwnerName: repo.OwnerName,
|
|
|
|
RepoName: repo.Name,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
2020-01-11 10:59:41 +08:00
|
|
|
// ToTrackedTimeList converts TrackedTimeList to API format
|
2022-06-13 17:37:59 +08:00
|
|
|
func ToTrackedTimeList(tl issues_model.TrackedTimeList) api.TrackedTimeList {
|
2020-01-11 10:59:41 +08:00
|
|
|
result := make([]*api.TrackedTime, 0, len(tl))
|
|
|
|
for _, t := range tl {
|
|
|
|
result = append(result, ToTrackedTime(t))
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
2020-02-29 03:49:50 +01:00
|
|
|
|
|
|
|
// ToLabel converts Label to API format
|
2022-06-13 17:37:59 +08:00
|
|
|
func ToLabel(label *issues_model.Label, repo *repo_model.Repository, org *user_model.User) *api.Label {
|
2021-09-10 18:03:16 +02:00
|
|
|
result := &api.Label{
|
2020-02-29 03:49:50 +01:00
|
|
|
ID: label.ID,
|
|
|
|
Name: label.Name,
|
|
|
|
Color: strings.TrimLeft(label.Color, "#"),
|
|
|
|
Description: label.Description,
|
|
|
|
}
|
2021-09-10 18:03:16 +02:00
|
|
|
|
|
|
|
// calculate URL
|
|
|
|
if label.BelongsToRepo() && repo != nil {
|
|
|
|
if repo != nil {
|
|
|
|
result.URL = fmt.Sprintf("%s/labels/%d", repo.APIURL(), label.ID)
|
|
|
|
} else {
|
|
|
|
log.Error("ToLabel did not get repo to calculate url for label with id '%d'", label.ID)
|
|
|
|
}
|
|
|
|
} else { // BelongsToOrg
|
|
|
|
if org != nil {
|
2021-11-16 18:18:25 +00:00
|
|
|
result.URL = fmt.Sprintf("%sapi/v1/orgs/%s/labels/%d", setting.AppURL, url.PathEscape(org.Name), label.ID)
|
2021-09-10 18:03:16 +02:00
|
|
|
} else {
|
|
|
|
log.Error("ToLabel did not get org to calculate url for label with id '%d'", label.ID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
2020-02-29 03:49:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// ToLabelList converts list of Label to API format
|
2022-06-13 17:37:59 +08:00
|
|
|
func ToLabelList(labels []*issues_model.Label, repo *repo_model.Repository, org *user_model.User) []*api.Label {
|
2020-02-29 03:49:50 +01:00
|
|
|
result := make([]*api.Label, len(labels))
|
|
|
|
for i := range labels {
|
2021-09-10 18:03:16 +02:00
|
|
|
result[i] = ToLabel(labels[i], repo, org)
|
2020-02-29 03:49:50 +01:00
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
2020-05-12 23:54:35 +02:00
|
|
|
|
|
|
|
// ToAPIMilestone converts Milestone into API Format
|
2022-04-08 17:11:15 +08:00
|
|
|
func ToAPIMilestone(m *issues_model.Milestone) *api.Milestone {
|
2020-05-12 23:54:35 +02:00
|
|
|
apiMilestone := &api.Milestone{
|
|
|
|
ID: m.ID,
|
|
|
|
State: m.State(),
|
|
|
|
Title: m.Name,
|
|
|
|
Description: m.Content,
|
|
|
|
OpenIssues: m.NumOpenIssues,
|
|
|
|
ClosedIssues: m.NumClosedIssues,
|
2020-09-05 19:38:54 +02:00
|
|
|
Created: m.CreatedUnix.AsTime(),
|
|
|
|
Updated: m.UpdatedUnix.AsTimePtr(),
|
2020-05-12 23:54:35 +02:00
|
|
|
}
|
|
|
|
if m.IsClosed {
|
|
|
|
apiMilestone.Closed = m.ClosedDateUnix.AsTimePtr()
|
|
|
|
}
|
|
|
|
if m.DeadlineUnix.Year() < 9999 {
|
|
|
|
apiMilestone.Deadline = m.DeadlineUnix.AsTimePtr()
|
|
|
|
}
|
|
|
|
return apiMilestone
|
|
|
|
}
|