mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 16:40:24 +08:00 
			
		
		
		
	Include thread related headers in issue/coment mail (#7484)
* Include thread related headers in issue/coment mail Make it so mail programs will group comments from an issue into the same thread by setting Message-ID on initial issue and then using In-Reply-To and References headers to reference that later on. * Add tests * more tests * fix typo
This commit is contained in:
		
				
					committed by
					
						
						techknowlogick
					
				
			
			
				
	
			
			
			
						parent
						
							5d3e351864
						
					
				
				
					commit
					944d904980
				
			@@ -472,6 +472,18 @@ func (issue *Issue) sendLabelUpdatedWebhook(doer *User) {
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ReplyReference returns tokenized address to use for email reply headers
 | 
				
			||||||
 | 
					func (issue *Issue) ReplyReference() string {
 | 
				
			||||||
 | 
						var path string
 | 
				
			||||||
 | 
						if issue.IsPull {
 | 
				
			||||||
 | 
							path = "pulls"
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							path = "issues"
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return fmt.Sprintf("%s/%s/%d@%s", issue.Repo.FullName(), path, issue.Index, setting.Domain)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (issue *Issue) addLabel(e *xorm.Session, label *Label, doer *User) error {
 | 
					func (issue *Issue) addLabel(e *xorm.Session, label *Label, doer *User) error {
 | 
				
			||||||
	return newIssueLabel(e, issue, label, doer)
 | 
						return newIssueLabel(e, issue, label, doer)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -156,7 +156,13 @@ func composeTplData(subject, body, link string) map[string]interface{} {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func composeIssueCommentMessage(issue *Issue, doer *User, content string, comment *Comment, tplName base.TplName, tos []string, info string) *mailer.Message {
 | 
					func composeIssueCommentMessage(issue *Issue, doer *User, content string, comment *Comment, tplName base.TplName, tos []string, info string) *mailer.Message {
 | 
				
			||||||
	subject := issue.mailSubject()
 | 
						var subject string
 | 
				
			||||||
 | 
						if comment != nil {
 | 
				
			||||||
 | 
							subject = "Re: " + issue.mailSubject()
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							subject = issue.mailSubject()
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err := issue.LoadRepo()
 | 
						err := issue.LoadRepo()
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		log.Error("LoadRepo: %v", err)
 | 
							log.Error("LoadRepo: %v", err)
 | 
				
			||||||
@@ -179,6 +185,15 @@ func composeIssueCommentMessage(issue *Issue, doer *User, content string, commen
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	msg := mailer.NewMessageFrom(tos, doer.DisplayName(), setting.MailService.FromEmail, subject, mailBody.String())
 | 
						msg := mailer.NewMessageFrom(tos, doer.DisplayName(), setting.MailService.FromEmail, subject, mailBody.String())
 | 
				
			||||||
	msg.Info = fmt.Sprintf("Subject: %s, %s", subject, info)
 | 
						msg.Info = fmt.Sprintf("Subject: %s, %s", subject, info)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Set Message-ID on first message so replies know what to reference
 | 
				
			||||||
 | 
						if comment == nil {
 | 
				
			||||||
 | 
							msg.SetHeader("Message-ID", "<"+issue.ReplyReference()+">")
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							msg.SetHeader("In-Reply-To", "<"+issue.ReplyReference()+">")
 | 
				
			||||||
 | 
							msg.SetHeader("References", "<"+issue.ReplyReference()+">")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return msg
 | 
						return msg
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										87
									
								
								models/mail_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								models/mail_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,87 @@
 | 
				
			|||||||
 | 
					// Copyright 2019 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 (
 | 
				
			||||||
 | 
						"html/template"
 | 
				
			||||||
 | 
						"testing"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"code.gitea.io/gitea/modules/setting"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						"github.com/stretchr/testify/assert"
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const tmpl = `
 | 
				
			||||||
 | 
					<!DOCTYPE html>
 | 
				
			||||||
 | 
					<html>
 | 
				
			||||||
 | 
					<head>
 | 
				
			||||||
 | 
						<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
 | 
				
			||||||
 | 
						<title>{{.Subject}}</title>
 | 
				
			||||||
 | 
					</head>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<body>
 | 
				
			||||||
 | 
						<p>{{.Body}}</p>
 | 
				
			||||||
 | 
						<p>
 | 
				
			||||||
 | 
							---
 | 
				
			||||||
 | 
							<br>
 | 
				
			||||||
 | 
							<a href="{{.Link}}">View it on Gitea</a>.
 | 
				
			||||||
 | 
						</p>
 | 
				
			||||||
 | 
					</body>
 | 
				
			||||||
 | 
					</html>
 | 
				
			||||||
 | 
					`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestComposeIssueCommentMessage(t *testing.T) {
 | 
				
			||||||
 | 
						assert.NoError(t, PrepareTestDatabase())
 | 
				
			||||||
 | 
						var MailService setting.Mailer
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						MailService.From = "test@gitea.com"
 | 
				
			||||||
 | 
						setting.MailService = &MailService
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						doer := AssertExistsAndLoadBean(t, &User{ID: 2}).(*User)
 | 
				
			||||||
 | 
						repo := AssertExistsAndLoadBean(t, &Repository{ID: 1, Owner: doer}).(*Repository)
 | 
				
			||||||
 | 
						issue := AssertExistsAndLoadBean(t, &Issue{ID: 1, Repo: repo, Poster: doer}).(*Issue)
 | 
				
			||||||
 | 
						comment := AssertExistsAndLoadBean(t, &Comment{ID: 2, Issue: issue}).(*Comment)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						email := template.Must(template.New("issue/comment").Parse(tmpl))
 | 
				
			||||||
 | 
						InitMailRender(email)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tos := []string{"test@gitea.com", "test2@gitea.com"}
 | 
				
			||||||
 | 
						msg := composeIssueCommentMessage(issue, doer, "test body", comment, mailIssueComment, tos, "issue comment")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						subject := msg.GetHeader("Subject")
 | 
				
			||||||
 | 
						inreplyTo := msg.GetHeader("In-Reply-To")
 | 
				
			||||||
 | 
						references := msg.GetHeader("References")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						assert.Equal(t, subject[0], "Re: "+issue.mailSubject(), "Comment reply subject should contain Re:")
 | 
				
			||||||
 | 
						assert.Equal(t, inreplyTo[0], "<user2/repo1/issues/1@localhost>", "In-Reply-To header doesn't match")
 | 
				
			||||||
 | 
						assert.Equal(t, references[0], "<user2/repo1/issues/1@localhost>", "References header doesn't match")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestComposeIssueMessage(t *testing.T) {
 | 
				
			||||||
 | 
						assert.NoError(t, PrepareTestDatabase())
 | 
				
			||||||
 | 
						var MailService setting.Mailer
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						MailService.From = "test@gitea.com"
 | 
				
			||||||
 | 
						setting.MailService = &MailService
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						doer := AssertExistsAndLoadBean(t, &User{ID: 2}).(*User)
 | 
				
			||||||
 | 
						repo := AssertExistsAndLoadBean(t, &Repository{ID: 1, Owner: doer}).(*Repository)
 | 
				
			||||||
 | 
						issue := AssertExistsAndLoadBean(t, &Issue{ID: 1, Repo: repo, Poster: doer}).(*Issue)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						email := template.Must(template.New("issue/comment").Parse(tmpl))
 | 
				
			||||||
 | 
						InitMailRender(email)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tos := []string{"test@gitea.com", "test2@gitea.com"}
 | 
				
			||||||
 | 
						msg := composeIssueCommentMessage(issue, doer, "test body", nil, mailIssueComment, tos, "issue create")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						subject := msg.GetHeader("Subject")
 | 
				
			||||||
 | 
						messageID := msg.GetHeader("Message-ID")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						assert.Equal(t, subject[0], issue.mailSubject(), "Subject not equal to issue.mailSubject()")
 | 
				
			||||||
 | 
						assert.Nil(t, msg.GetHeader("In-Reply-To"))
 | 
				
			||||||
 | 
						assert.Nil(t, msg.GetHeader("References"))
 | 
				
			||||||
 | 
						assert.Equal(t, messageID[0], "<user2/repo1/issues/1@localhost>", "Message-ID header doesn't match")
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
		Reference in New Issue
	
	Block a user