#1597 fix activitity feeds for pull requests
This commit is contained in:
parent
a2f13eae55
commit
414eb22ef9
10 changed files with 338 additions and 278 deletions
|
@ -3,7 +3,7 @@ Gogs - Go Git Service [![Build Status](https://travis-ci.org/gogits/gogs.svg?bra
|
||||||
|
|
||||||
![](https://github.com/gogits/gogs/blob/master/public/img/gogs-large-resize.png?raw=true)
|
![](https://github.com/gogits/gogs/blob/master/public/img/gogs-large-resize.png?raw=true)
|
||||||
|
|
||||||
##### Current version: 0.8.57
|
##### Current version: 0.8.58
|
||||||
|
|
||||||
| Web | UI | Preview |
|
| Web | UI | Preview |
|
||||||
|:-------------:|:-------:|:-------:|
|
|:-------------:|:-------:|:-------:|
|
||||||
|
|
|
@ -1064,6 +1064,8 @@ create_issue = `opened issue <a href="%s/issues/%s">%s#%[2]s</a>`
|
||||||
close_issue = `closed issue <a href="%s/issues/%s">%s#%[2]s</a>`
|
close_issue = `closed issue <a href="%s/issues/%s">%s#%[2]s</a>`
|
||||||
reopen_issue = `reopened issue <a href="%s/issues/%s">%s#%[2]s</a>`
|
reopen_issue = `reopened issue <a href="%s/issues/%s">%s#%[2]s</a>`
|
||||||
create_pull_request = `created pull request <a href="%s/pulls/%s">%s#%[2]s</a>`
|
create_pull_request = `created pull request <a href="%s/pulls/%s">%s#%[2]s</a>`
|
||||||
|
close_pull_request = `closed pull request <a href="%s/pulls/%s">%s#%[2]s</a>`
|
||||||
|
reopen_pull_request = `reopened pull request <a href="%s/pulls/%s">%s#%[2]s</a>`
|
||||||
comment_issue = `commented on issue <a href="%s/issues/%s">%s#%[2]s</a>`
|
comment_issue = `commented on issue <a href="%s/issues/%s">%s#%[2]s</a>`
|
||||||
merge_pull_request = `merged pull request <a href="%s/pulls/%s">%s#%[2]s</a>`
|
merge_pull_request = `merged pull request <a href="%s/pulls/%s">%s#%[2]s</a>`
|
||||||
transfer_repo = transfered repository <code>%s</code> to <a href="%s">%s</a>
|
transfer_repo = transfered repository <code>%s</code> to <a href="%s">%s</a>
|
||||||
|
|
2
gogs.go
2
gogs.go
|
@ -17,7 +17,7 @@ import (
|
||||||
"github.com/gogits/gogs/modules/setting"
|
"github.com/gogits/gogs/modules/setting"
|
||||||
)
|
)
|
||||||
|
|
||||||
const APP_VER = "0.8.57.0305"
|
const APP_VER = "0.8.58.0305"
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
runtime.GOMAXPROCS(runtime.NumCPU())
|
runtime.GOMAXPROCS(runtime.NumCPU())
|
||||||
|
|
|
@ -41,6 +41,8 @@ const (
|
||||||
ACTION_MERGE_PULL_REQUEST // 11
|
ACTION_MERGE_PULL_REQUEST // 11
|
||||||
ACTION_CLOSE_ISSUE // 12
|
ACTION_CLOSE_ISSUE // 12
|
||||||
ACTION_REOPEN_ISSUE // 13
|
ACTION_REOPEN_ISSUE // 13
|
||||||
|
ACTION_CLOSE_PULL_REQUEST // 14
|
||||||
|
ACTION_REOPEN_PULL_REQUEST // 15
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
|
275
models/issue.go
275
models/issue.go
|
@ -219,6 +219,7 @@ func (i *Issue) ReadBy(uid int64) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *Issue) changeStatus(e *xorm.Session, doer *User, repo *Repository, isClosed bool) (err error) {
|
func (i *Issue) changeStatus(e *xorm.Session, doer *User, repo *Repository, isClosed bool) (err error) {
|
||||||
|
// Nothing should be performed if current status is same as target status
|
||||||
if i.IsClosed == isClosed {
|
if i.IsClosed == isClosed {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -230,7 +231,7 @@ func (i *Issue) changeStatus(e *xorm.Session, doer *User, repo *Repository, isCl
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update labels.
|
// Update issue count of labels
|
||||||
if err = i.getLabels(e); err != nil {
|
if err = i.getLabels(e); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -245,12 +246,12 @@ func (i *Issue) changeStatus(e *xorm.Session, doer *User, repo *Repository, isCl
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update milestone.
|
// Update issue count of milestone
|
||||||
if err = changeMilestoneIssueStats(e, i); err != nil {
|
if err = changeMilestoneIssueStats(e, i); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// New action comment.
|
// New action comment
|
||||||
if _, err = createStatusComment(e, doer, repo, i); err != nil {
|
if _, err = createStatusComment(e, doer, repo, i); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -258,7 +259,7 @@ func (i *Issue) changeStatus(e *xorm.Session, doer *User, repo *Repository, isCl
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChangeStatus changes issue status to open/closed.
|
// ChangeStatus changes issue status to open or closed.
|
||||||
func (i *Issue) ChangeStatus(doer *User, repo *Repository, isClosed bool) (err error) {
|
func (i *Issue) ChangeStatus(doer *User, repo *Repository, isClosed bool) (err error) {
|
||||||
sess := x.NewSession()
|
sess := x.NewSession()
|
||||||
defer sessionRelease(sess)
|
defer sessionRelease(sess)
|
||||||
|
@ -857,7 +858,7 @@ func UpdateIssue(issue *Issue) error {
|
||||||
return updateIssue(x, issue)
|
return updateIssue(x, issue)
|
||||||
}
|
}
|
||||||
|
|
||||||
// updateIssueCols updates specific fields of given issue.
|
// updateIssueCols only updates values of specific columns for given issue.
|
||||||
func updateIssueCols(e Engine, issue *Issue, cols ...string) error {
|
func updateIssueCols(e Engine, issue *Issue, cols ...string) error {
|
||||||
_, err := e.Id(issue.ID).Cols(cols...).Update(issue)
|
_, err := e.Id(issue.ID).Cols(cols...).Update(issue)
|
||||||
return err
|
return err
|
||||||
|
@ -1241,270 +1242,6 @@ func DeleteMilestoneByID(id int64) error {
|
||||||
return sess.Commit()
|
return sess.Commit()
|
||||||
}
|
}
|
||||||
|
|
||||||
// _________ __
|
|
||||||
// \_ ___ \ ____ _____ _____ ____ _____/ |_
|
|
||||||
// / \ \/ / _ \ / \ / \_/ __ \ / \ __\
|
|
||||||
// \ \___( <_> ) Y Y \ Y Y \ ___/| | \ |
|
|
||||||
// \______ /\____/|__|_| /__|_| /\___ >___| /__|
|
|
||||||
// \/ \/ \/ \/ \/
|
|
||||||
|
|
||||||
// CommentType defines whether a comment is just a simple comment, an action (like close) or a reference.
|
|
||||||
type CommentType int
|
|
||||||
|
|
||||||
const (
|
|
||||||
// Plain comment, can be associated with a commit (CommitId > 0) and a line (Line > 0)
|
|
||||||
COMMENT_TYPE_COMMENT CommentType = iota
|
|
||||||
COMMENT_TYPE_REOPEN
|
|
||||||
COMMENT_TYPE_CLOSE
|
|
||||||
|
|
||||||
// References.
|
|
||||||
COMMENT_TYPE_ISSUE_REF
|
|
||||||
// Reference from a commit (not part of a pull request)
|
|
||||||
COMMENT_TYPE_COMMIT_REF
|
|
||||||
// Reference from a comment
|
|
||||||
COMMENT_TYPE_COMMENT_REF
|
|
||||||
// Reference from a pull request
|
|
||||||
COMMENT_TYPE_PULL_REF
|
|
||||||
)
|
|
||||||
|
|
||||||
type CommentTag int
|
|
||||||
|
|
||||||
const (
|
|
||||||
COMMENT_TAG_NONE CommentTag = iota
|
|
||||||
COMMENT_TAG_POSTER
|
|
||||||
COMMENT_TAG_ADMIN
|
|
||||||
COMMENT_TAG_OWNER
|
|
||||||
)
|
|
||||||
|
|
||||||
// Comment represents a comment in commit and issue page.
|
|
||||||
type Comment struct {
|
|
||||||
ID int64 `xorm:"pk autoincr"`
|
|
||||||
Type CommentType
|
|
||||||
PosterID int64
|
|
||||||
Poster *User `xorm:"-"`
|
|
||||||
IssueID int64 `xorm:"INDEX"`
|
|
||||||
CommitID int64
|
|
||||||
Line int64
|
|
||||||
Content string `xorm:"TEXT"`
|
|
||||||
RenderedContent string `xorm:"-"`
|
|
||||||
Created time.Time `xorm:"CREATED"`
|
|
||||||
|
|
||||||
// Reference issue in commit message
|
|
||||||
CommitSHA string `xorm:"VARCHAR(40)"`
|
|
||||||
|
|
||||||
Attachments []*Attachment `xorm:"-"`
|
|
||||||
|
|
||||||
// For view issue page.
|
|
||||||
ShowTag CommentTag `xorm:"-"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Comment) AfterSet(colName string, _ xorm.Cell) {
|
|
||||||
var err error
|
|
||||||
switch colName {
|
|
||||||
case "id":
|
|
||||||
c.Attachments, err = GetAttachmentsByCommentID(c.ID)
|
|
||||||
if err != nil {
|
|
||||||
log.Error(3, "GetAttachmentsByCommentID[%d]: %v", c.ID, err)
|
|
||||||
}
|
|
||||||
|
|
||||||
case "poster_id":
|
|
||||||
c.Poster, err = GetUserByID(c.PosterID)
|
|
||||||
if err != nil {
|
|
||||||
if IsErrUserNotExist(err) {
|
|
||||||
c.PosterID = -1
|
|
||||||
c.Poster = NewFakeUser()
|
|
||||||
} else {
|
|
||||||
log.Error(3, "GetUserByID[%d]: %v", c.ID, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case "created":
|
|
||||||
c.Created = regulateTimeZone(c.Created)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Comment) AfterDelete() {
|
|
||||||
_, err := DeleteAttachmentsByComment(c.ID, true)
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
log.Info("Could not delete files for comment %d on issue #%d: %s", c.ID, c.IssueID, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// HashTag returns unique hash tag for comment.
|
|
||||||
func (c *Comment) HashTag() string {
|
|
||||||
return "issuecomment-" + com.ToStr(c.ID)
|
|
||||||
}
|
|
||||||
|
|
||||||
// EventTag returns unique event hash tag for comment.
|
|
||||||
func (c *Comment) EventTag() string {
|
|
||||||
return "event-" + com.ToStr(c.ID)
|
|
||||||
}
|
|
||||||
|
|
||||||
func createComment(e *xorm.Session, u *User, repo *Repository, issue *Issue, commitID, line int64, cmtType CommentType, content, commitSHA string, uuids []string) (_ *Comment, err error) {
|
|
||||||
comment := &Comment{
|
|
||||||
PosterID: u.Id,
|
|
||||||
Type: cmtType,
|
|
||||||
IssueID: issue.ID,
|
|
||||||
CommitID: commitID,
|
|
||||||
Line: line,
|
|
||||||
Content: content,
|
|
||||||
CommitSHA: commitSHA,
|
|
||||||
}
|
|
||||||
if _, err = e.Insert(comment); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Compose comment action, could be plain comment, close or reopen issue.
|
|
||||||
// This object will be used to notify watchers in the end of function.
|
|
||||||
act := &Action{
|
|
||||||
ActUserID: u.Id,
|
|
||||||
ActUserName: u.Name,
|
|
||||||
ActEmail: u.Email,
|
|
||||||
Content: fmt.Sprintf("%d|%s", issue.Index, strings.Split(content, "\n")[0]),
|
|
||||||
RepoID: repo.ID,
|
|
||||||
RepoUserName: repo.Owner.Name,
|
|
||||||
RepoName: repo.Name,
|
|
||||||
IsPrivate: repo.IsPrivate,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check comment type.
|
|
||||||
switch cmtType {
|
|
||||||
case COMMENT_TYPE_COMMENT:
|
|
||||||
act.OpType = ACTION_COMMENT_ISSUE
|
|
||||||
|
|
||||||
if _, err = e.Exec("UPDATE `issue` SET num_comments=num_comments+1 WHERE id=?", issue.ID); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check attachments.
|
|
||||||
attachments := make([]*Attachment, 0, len(uuids))
|
|
||||||
for _, uuid := range uuids {
|
|
||||||
attach, err := getAttachmentByUUID(e, uuid)
|
|
||||||
if err != nil {
|
|
||||||
if IsErrAttachmentNotExist(err) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
return nil, fmt.Errorf("getAttachmentByUUID[%s]: %v", uuid, err)
|
|
||||||
}
|
|
||||||
attachments = append(attachments, attach)
|
|
||||||
}
|
|
||||||
|
|
||||||
for i := range attachments {
|
|
||||||
attachments[i].IssueID = issue.ID
|
|
||||||
attachments[i].CommentID = comment.ID
|
|
||||||
// No assign value could be 0, so ignore AllCols().
|
|
||||||
if _, err = e.Id(attachments[i].ID).Update(attachments[i]); err != nil {
|
|
||||||
return nil, fmt.Errorf("update attachment[%d]: %v", attachments[i].ID, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
case COMMENT_TYPE_REOPEN:
|
|
||||||
act.OpType = ACTION_REOPEN_ISSUE
|
|
||||||
|
|
||||||
if issue.IsPull {
|
|
||||||
_, err = e.Exec("UPDATE `repository` SET num_closed_pulls=num_closed_pulls-1 WHERE id=?", repo.ID)
|
|
||||||
} else {
|
|
||||||
_, err = e.Exec("UPDATE `repository` SET num_closed_issues=num_closed_issues-1 WHERE id=?", repo.ID)
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
case COMMENT_TYPE_CLOSE:
|
|
||||||
act.OpType = ACTION_CLOSE_ISSUE
|
|
||||||
|
|
||||||
if issue.IsPull {
|
|
||||||
_, err = e.Exec("UPDATE `repository` SET num_closed_pulls=num_closed_pulls+1 WHERE id=?", repo.ID)
|
|
||||||
} else {
|
|
||||||
_, err = e.Exec("UPDATE `repository` SET num_closed_issues=num_closed_issues+1 WHERE id=?", repo.ID)
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Notify watchers for whatever action comes in.
|
|
||||||
if err = notifyWatchers(e, act); err != nil {
|
|
||||||
return nil, fmt.Errorf("notifyWatchers: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
return comment, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func createStatusComment(e *xorm.Session, doer *User, repo *Repository, issue *Issue) (*Comment, error) {
|
|
||||||
cmtType := COMMENT_TYPE_CLOSE
|
|
||||||
if !issue.IsClosed {
|
|
||||||
cmtType = COMMENT_TYPE_REOPEN
|
|
||||||
}
|
|
||||||
return createComment(e, doer, repo, issue, 0, 0, cmtType, "", "", nil)
|
|
||||||
}
|
|
||||||
|
|
||||||
// CreateComment creates comment of issue or commit.
|
|
||||||
func CreateComment(doer *User, repo *Repository, issue *Issue, commitID, line int64, cmtType CommentType, content, commitSHA string, attachments []string) (comment *Comment, err error) {
|
|
||||||
sess := x.NewSession()
|
|
||||||
defer sessionRelease(sess)
|
|
||||||
if err = sess.Begin(); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
comment, err = createComment(sess, doer, repo, issue, commitID, line, cmtType, content, commitSHA, attachments)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return comment, sess.Commit()
|
|
||||||
}
|
|
||||||
|
|
||||||
// CreateIssueComment creates a plain issue comment.
|
|
||||||
func CreateIssueComment(doer *User, repo *Repository, issue *Issue, content string, attachments []string) (*Comment, error) {
|
|
||||||
return CreateComment(doer, repo, issue, 0, 0, COMMENT_TYPE_COMMENT, content, "", attachments)
|
|
||||||
}
|
|
||||||
|
|
||||||
// CreateRefComment creates a commit reference comment to issue.
|
|
||||||
func CreateRefComment(doer *User, repo *Repository, issue *Issue, content, commitSHA string) error {
|
|
||||||
if len(commitSHA) == 0 {
|
|
||||||
return fmt.Errorf("cannot create reference with empty commit SHA")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if same reference from same commit has already existed.
|
|
||||||
has, err := x.Get(&Comment{
|
|
||||||
Type: COMMENT_TYPE_COMMIT_REF,
|
|
||||||
IssueID: issue.ID,
|
|
||||||
CommitSHA: commitSHA,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("check reference comment: %v", err)
|
|
||||||
} else if has {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
_, err = CreateComment(doer, repo, issue, 0, 0, COMMENT_TYPE_COMMIT_REF, content, commitSHA, nil)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetCommentByID returns the comment by given ID.
|
|
||||||
func GetCommentByID(id int64) (*Comment, error) {
|
|
||||||
c := new(Comment)
|
|
||||||
has, err := x.Id(id).Get(c)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
} else if !has {
|
|
||||||
return nil, ErrCommentNotExist{id}
|
|
||||||
}
|
|
||||||
return c, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetCommentsByIssueID returns all comments of issue by given ID.
|
|
||||||
func GetCommentsByIssueID(issueID int64) ([]*Comment, error) {
|
|
||||||
comments := make([]*Comment, 0, 10)
|
|
||||||
return comments, x.Where("issue_id=?", issueID).Asc("created").Find(&comments)
|
|
||||||
}
|
|
||||||
|
|
||||||
// UpdateComment updates information of comment.
|
|
||||||
func UpdateComment(c *Comment) error {
|
|
||||||
_, err := x.Id(c.ID).AllCols().Update(c)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Attachment represent a attachment of issue/comment/release.
|
// Attachment represent a attachment of issue/comment/release.
|
||||||
type Attachment struct {
|
type Attachment struct {
|
||||||
ID int64 `xorm:"pk autoincr"`
|
ID int64 `xorm:"pk autoincr"`
|
||||||
|
|
311
models/issue_comment.go
Normal file
311
models/issue_comment.go
Normal file
|
@ -0,0 +1,311 @@
|
||||||
|
// Copyright 2016 The Gogs 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"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/Unknwon/com"
|
||||||
|
"github.com/go-xorm/xorm"
|
||||||
|
|
||||||
|
"github.com/gogits/gogs/modules/log"
|
||||||
|
)
|
||||||
|
|
||||||
|
// CommentType defines whether a comment is just a simple comment, an action (like close) or a reference.
|
||||||
|
type CommentType int
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Plain comment, can be associated with a commit (CommitID > 0) and a line (LineNum > 0)
|
||||||
|
COMMENT_TYPE_COMMENT CommentType = iota
|
||||||
|
COMMENT_TYPE_REOPEN
|
||||||
|
COMMENT_TYPE_CLOSE
|
||||||
|
|
||||||
|
// References.
|
||||||
|
COMMENT_TYPE_ISSUE_REF
|
||||||
|
// Reference from a commit (not part of a pull request)
|
||||||
|
COMMENT_TYPE_COMMIT_REF
|
||||||
|
// Reference from a comment
|
||||||
|
COMMENT_TYPE_COMMENT_REF
|
||||||
|
// Reference from a pull request
|
||||||
|
COMMENT_TYPE_PULL_REF
|
||||||
|
)
|
||||||
|
|
||||||
|
type CommentTag int
|
||||||
|
|
||||||
|
const (
|
||||||
|
COMMENT_TAG_NONE CommentTag = iota
|
||||||
|
COMMENT_TAG_POSTER
|
||||||
|
COMMENT_TAG_ADMIN
|
||||||
|
COMMENT_TAG_OWNER
|
||||||
|
)
|
||||||
|
|
||||||
|
// Comment represents a comment in commit and issue page.
|
||||||
|
type Comment struct {
|
||||||
|
ID int64 `xorm:"pk autoincr"`
|
||||||
|
Type CommentType
|
||||||
|
PosterID int64
|
||||||
|
Poster *User `xorm:"-"`
|
||||||
|
IssueID int64 `xorm:"INDEX"`
|
||||||
|
CommitID int64
|
||||||
|
Line int64
|
||||||
|
Content string `xorm:"TEXT"`
|
||||||
|
RenderedContent string `xorm:"-"`
|
||||||
|
Created time.Time `xorm:"CREATED"`
|
||||||
|
|
||||||
|
// Reference issue in commit message
|
||||||
|
CommitSHA string `xorm:"VARCHAR(40)"`
|
||||||
|
|
||||||
|
Attachments []*Attachment `xorm:"-"`
|
||||||
|
|
||||||
|
// For view issue page.
|
||||||
|
ShowTag CommentTag `xorm:"-"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Comment) AfterSet(colName string, _ xorm.Cell) {
|
||||||
|
var err error
|
||||||
|
switch colName {
|
||||||
|
case "id":
|
||||||
|
c.Attachments, err = GetAttachmentsByCommentID(c.ID)
|
||||||
|
if err != nil {
|
||||||
|
log.Error(3, "GetAttachmentsByCommentID[%d]: %v", c.ID, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
case "poster_id":
|
||||||
|
c.Poster, err = GetUserByID(c.PosterID)
|
||||||
|
if err != nil {
|
||||||
|
if IsErrUserNotExist(err) {
|
||||||
|
c.PosterID = -1
|
||||||
|
c.Poster = NewFakeUser()
|
||||||
|
} else {
|
||||||
|
log.Error(3, "GetUserByID[%d]: %v", c.ID, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case "created":
|
||||||
|
c.Created = regulateTimeZone(c.Created)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Comment) AfterDelete() {
|
||||||
|
_, err := DeleteAttachmentsByComment(c.ID, true)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Info("Could not delete files for comment %d on issue #%d: %s", c.ID, c.IssueID, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HashTag returns unique hash tag for comment.
|
||||||
|
func (c *Comment) HashTag() string {
|
||||||
|
return "issuecomment-" + com.ToStr(c.ID)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EventTag returns unique event hash tag for comment.
|
||||||
|
func (c *Comment) EventTag() string {
|
||||||
|
return "event-" + com.ToStr(c.ID)
|
||||||
|
}
|
||||||
|
|
||||||
|
func createComment(e *xorm.Session, opts *CreateCommentOptions) (_ *Comment, err error) {
|
||||||
|
comment := &Comment{
|
||||||
|
Type: opts.Type,
|
||||||
|
PosterID: opts.Doer.Id,
|
||||||
|
IssueID: opts.Issue.ID,
|
||||||
|
CommitID: opts.CommitID,
|
||||||
|
CommitSHA: opts.CommitSHA,
|
||||||
|
Line: opts.LineNum,
|
||||||
|
Content: opts.Content,
|
||||||
|
}
|
||||||
|
if _, err = e.Insert(comment); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Compose comment action, could be plain comment, close or reopen issue.
|
||||||
|
// This object will be used to notify watchers in the end of function.
|
||||||
|
act := &Action{
|
||||||
|
ActUserID: opts.Doer.Id,
|
||||||
|
ActUserName: opts.Doer.Name,
|
||||||
|
ActEmail: opts.Doer.Email,
|
||||||
|
Content: fmt.Sprintf("%d|%s", opts.Issue.Index, strings.Split(opts.Content, "\n")[0]),
|
||||||
|
RepoID: opts.Repo.ID,
|
||||||
|
RepoUserName: opts.Repo.Owner.Name,
|
||||||
|
RepoName: opts.Repo.Name,
|
||||||
|
IsPrivate: opts.Repo.IsPrivate,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check comment type.
|
||||||
|
switch opts.Type {
|
||||||
|
case COMMENT_TYPE_COMMENT:
|
||||||
|
act.OpType = ACTION_COMMENT_ISSUE
|
||||||
|
|
||||||
|
if _, err = e.Exec("UPDATE `issue` SET num_comments=num_comments+1 WHERE id=?", opts.Issue.ID); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check attachments
|
||||||
|
attachments := make([]*Attachment, 0, len(opts.Attachments))
|
||||||
|
for _, uuid := range opts.Attachments {
|
||||||
|
attach, err := getAttachmentByUUID(e, uuid)
|
||||||
|
if err != nil {
|
||||||
|
if IsErrAttachmentNotExist(err) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("getAttachmentByUUID[%s]: %v", uuid, err)
|
||||||
|
}
|
||||||
|
attachments = append(attachments, attach)
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := range attachments {
|
||||||
|
attachments[i].IssueID = opts.Issue.ID
|
||||||
|
attachments[i].CommentID = comment.ID
|
||||||
|
// No assign value could be 0, so ignore AllCols().
|
||||||
|
if _, err = e.Id(attachments[i].ID).Update(attachments[i]); err != nil {
|
||||||
|
return nil, fmt.Errorf("update attachment[%d]: %v", attachments[i].ID, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
case COMMENT_TYPE_REOPEN:
|
||||||
|
act.OpType = ACTION_REOPEN_ISSUE
|
||||||
|
if opts.Issue.IsPull {
|
||||||
|
act.OpType = ACTION_REOPEN_PULL_REQUEST
|
||||||
|
}
|
||||||
|
|
||||||
|
if opts.Issue.IsPull {
|
||||||
|
_, err = e.Exec("UPDATE `repository` SET num_closed_pulls=num_closed_pulls-1 WHERE id=?", opts.Repo.ID)
|
||||||
|
} else {
|
||||||
|
_, err = e.Exec("UPDATE `repository` SET num_closed_issues=num_closed_issues-1 WHERE id=?", opts.Repo.ID)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
case COMMENT_TYPE_CLOSE:
|
||||||
|
act.OpType = ACTION_CLOSE_ISSUE
|
||||||
|
if opts.Issue.IsPull {
|
||||||
|
act.OpType = ACTION_CLOSE_PULL_REQUEST
|
||||||
|
}
|
||||||
|
|
||||||
|
if opts.Issue.IsPull {
|
||||||
|
_, err = e.Exec("UPDATE `repository` SET num_closed_pulls=num_closed_pulls+1 WHERE id=?", opts.Repo.ID)
|
||||||
|
} else {
|
||||||
|
_, err = e.Exec("UPDATE `repository` SET num_closed_issues=num_closed_issues+1 WHERE id=?", opts.Repo.ID)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Notify watchers for whatever action comes in
|
||||||
|
if err = notifyWatchers(e, act); err != nil {
|
||||||
|
return nil, fmt.Errorf("notifyWatchers: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return comment, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func createStatusComment(e *xorm.Session, doer *User, repo *Repository, issue *Issue) (*Comment, error) {
|
||||||
|
cmtType := COMMENT_TYPE_CLOSE
|
||||||
|
if !issue.IsClosed {
|
||||||
|
cmtType = COMMENT_TYPE_REOPEN
|
||||||
|
}
|
||||||
|
return createComment(e, &CreateCommentOptions{
|
||||||
|
Type: cmtType,
|
||||||
|
Doer: doer,
|
||||||
|
Repo: repo,
|
||||||
|
Issue: issue,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
type CreateCommentOptions struct {
|
||||||
|
Type CommentType
|
||||||
|
Doer *User
|
||||||
|
Repo *Repository
|
||||||
|
Issue *Issue
|
||||||
|
|
||||||
|
CommitID int64
|
||||||
|
CommitSHA string
|
||||||
|
LineNum int64
|
||||||
|
Content string
|
||||||
|
Attachments []string // UUIDs of attachments
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateComment creates comment of issue or commit.
|
||||||
|
func CreateComment(opts *CreateCommentOptions) (comment *Comment, err error) {
|
||||||
|
sess := x.NewSession()
|
||||||
|
defer sessionRelease(sess)
|
||||||
|
if err = sess.Begin(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
comment, err = createComment(sess, opts)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return comment, sess.Commit()
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateIssueComment creates a plain issue comment.
|
||||||
|
func CreateIssueComment(doer *User, repo *Repository, issue *Issue, content string, attachments []string) (*Comment, error) {
|
||||||
|
return CreateComment(&CreateCommentOptions{
|
||||||
|
Type: COMMENT_TYPE_COMMENT,
|
||||||
|
Doer: doer,
|
||||||
|
Repo: repo,
|
||||||
|
Issue: issue,
|
||||||
|
Content: content,
|
||||||
|
Attachments: attachments,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateRefComment creates a commit reference comment to issue.
|
||||||
|
func CreateRefComment(doer *User, repo *Repository, issue *Issue, content, commitSHA string) error {
|
||||||
|
if len(commitSHA) == 0 {
|
||||||
|
return fmt.Errorf("cannot create reference with empty commit SHA")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if same reference from same commit has already existed.
|
||||||
|
has, err := x.Get(&Comment{
|
||||||
|
Type: COMMENT_TYPE_COMMIT_REF,
|
||||||
|
IssueID: issue.ID,
|
||||||
|
CommitSHA: commitSHA,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("check reference comment: %v", err)
|
||||||
|
} else if has {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = CreateComment(&CreateCommentOptions{
|
||||||
|
Type: COMMENT_TYPE_COMMIT_REF,
|
||||||
|
Doer: doer,
|
||||||
|
Repo: repo,
|
||||||
|
Issue: issue,
|
||||||
|
CommitSHA: commitSHA,
|
||||||
|
Content: content,
|
||||||
|
})
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetCommentByID returns the comment by given ID.
|
||||||
|
func GetCommentByID(id int64) (*Comment, error) {
|
||||||
|
c := new(Comment)
|
||||||
|
has, err := x.Id(id).Get(c)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
} else if !has {
|
||||||
|
return nil, ErrCommentNotExist{id}
|
||||||
|
}
|
||||||
|
return c, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetCommentsByIssueID returns all comments of issue by given ID.
|
||||||
|
func GetCommentsByIssueID(issueID int64) ([]*Comment, error) {
|
||||||
|
comments := make([]*Comment, 0, 10)
|
||||||
|
return comments, x.Where("issue_id=?", issueID).Asc("created").Find(&comments)
|
||||||
|
}
|
||||||
|
|
||||||
|
// UpdateComment updates information of comment.
|
||||||
|
func UpdateComment(c *Comment) error {
|
||||||
|
_, err := x.Id(c.ID).AllCols().Update(c)
|
||||||
|
return err
|
||||||
|
}
|
File diff suppressed because one or more lines are too long
|
@ -233,7 +233,7 @@ func ActionIcon(opType int) string {
|
||||||
return "repo"
|
return "repo"
|
||||||
case 5, 9: // Commit repository
|
case 5, 9: // Commit repository
|
||||||
return "git-commit"
|
return "git-commit"
|
||||||
case 6, 13: // Create and reopen issue
|
case 6: // Create issue
|
||||||
return "issue-opened"
|
return "issue-opened"
|
||||||
case 7: // New pull request
|
case 7: // New pull request
|
||||||
return "git-pull-request"
|
return "git-pull-request"
|
||||||
|
@ -241,8 +241,10 @@ func ActionIcon(opType int) string {
|
||||||
return "comment"
|
return "comment"
|
||||||
case 11: // Merge pull request
|
case 11: // Merge pull request
|
||||||
return "git-merge"
|
return "git-merge"
|
||||||
case 12: // Close issue
|
case 12, 14: // Close issue or pull request
|
||||||
return "issue-closed"
|
return "issue-closed"
|
||||||
|
case 13, 15: // Reopen issue or pull request
|
||||||
|
return "issue-reopened"
|
||||||
default:
|
default:
|
||||||
return "invalid type"
|
return "invalid type"
|
||||||
}
|
}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
0.8.57.0305
|
0.8.58.0305
|
|
@ -37,6 +37,12 @@
|
||||||
{{else if eq .GetOpType 13}}
|
{{else if eq .GetOpType 13}}
|
||||||
{{ $index := index .GetIssueInfos 0}}
|
{{ $index := index .GetIssueInfos 0}}
|
||||||
{{$.i18n.Tr "action.reopen_issue" .GetRepoLink $index .ShortRepoPath | Str2html}}
|
{{$.i18n.Tr "action.reopen_issue" .GetRepoLink $index .ShortRepoPath | Str2html}}
|
||||||
|
{{else if eq .GetOpType 14}}
|
||||||
|
{{ $index := index .GetIssueInfos 0}}
|
||||||
|
{{$.i18n.Tr "action.close_pull_request" .GetRepoLink $index .ShortRepoPath | Str2html}}
|
||||||
|
{{else if eq .GetOpType 15}}
|
||||||
|
{{ $index := index .GetIssueInfos 0}}
|
||||||
|
{{$.i18n.Tr "action.reopen_pull_request" .GetRepoLink $index .ShortRepoPath | Str2html}}
|
||||||
{{end}}
|
{{end}}
|
||||||
</p>
|
</p>
|
||||||
{{if eq .GetOpType 5}}
|
{{if eq .GetOpType 5}}
|
||||||
|
@ -55,13 +61,13 @@
|
||||||
{{else if eq .GetOpType 6}}
|
{{else if eq .GetOpType 6}}
|
||||||
<span class="text truncate issue title has-emoji">{{index .GetIssueInfos 1}}</span>
|
<span class="text truncate issue title has-emoji">{{index .GetIssueInfos 1}}</span>
|
||||||
{{else if eq .GetOpType 7}}
|
{{else if eq .GetOpType 7}}
|
||||||
<p class="text light grey has-emoji">{{index .GetIssueInfos 1}}</p>
|
<span class="text truncate issue title has-emoji">{{index .GetIssueInfos 1}}</span>
|
||||||
{{else if eq .GetOpType 10}}
|
{{else if eq .GetOpType 10}}
|
||||||
<span class="text truncate issue title has-emoji">{{.GetIssueTitle}}</span>
|
<span class="text truncate issue title has-emoji">{{.GetIssueTitle}}</span>
|
||||||
<p class="text light grey has-emoji">{{index .GetIssueInfos 1}}</p>
|
<p class="text light grey has-emoji">{{index .GetIssueInfos 1}}</p>
|
||||||
{{else if eq .GetOpType 11}}
|
{{else if eq .GetOpType 11}}
|
||||||
<p class="text light grey has-emoji">{{index .GetIssueInfos 1}}</p>
|
<p class="text light grey has-emoji">{{index .GetIssueInfos 1}}</p>
|
||||||
{{else if (or (eq .GetOpType 12) (eq .GetOpType 13))}}
|
{{else if (or (or (eq .GetOpType 12) (eq .GetOpType 13)) (or (eq .GetOpType 14) (eq .GetOpType 15)))}}
|
||||||
<span class="text truncate issue title has-emoji">{{.GetIssueTitle}}</span>
|
<span class="text truncate issue title has-emoji">{{.GetIssueTitle}}</span>
|
||||||
{{end}}
|
{{end}}
|
||||||
<p class="text italic light grey">{{TimeSince .GetCreate $.i18n.Lang}}</p>
|
<p class="text italic light grey">{{TimeSince .GetCreate $.i18n.Lang}}</p>
|
||||||
|
|
Reference in a new issue