forked from gitea/gitea
		
	#1597 fix activitity feeds for pull requests
This commit is contained in:
		
							parent
							
								
									a2f13eae55
								
							
						
					
					
						commit
						414eb22ef9
					
				| @ -3,7 +3,7 @@ Gogs - Go Git Service [ |  | ||||||
| 
 | 
 | ||||||
| ##### 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> | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Unknwon
						Unknwon