mirror of https://github.com/gogits/gogs.git
Unknwon
9 years ago
26 changed files with 436 additions and 345 deletions
@ -0,0 +1,81 @@ |
|||||||
|
// 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" |
||||||
|
|
||||||
|
"github.com/Unknwon/com" |
||||||
|
|
||||||
|
"github.com/gogits/gogs/modules/log" |
||||||
|
"github.com/gogits/gogs/modules/markdown" |
||||||
|
"github.com/gogits/gogs/modules/setting" |
||||||
|
) |
||||||
|
|
||||||
|
func (issue *Issue) MailSubject() string { |
||||||
|
return fmt.Sprintf("[%s] %s (#%d)", issue.Repo.Name, issue.Name, issue.Index) |
||||||
|
} |
||||||
|
|
||||||
|
// mailIssueCommentToParticipants can be used for both new issue creation and comment.
|
||||||
|
func mailIssueCommentToParticipants(issue *Issue, doer *User, mentions []string) error { |
||||||
|
if !setting.Service.EnableNotifyMail { |
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
// Mail wahtcers.
|
||||||
|
watchers, err := GetWatchers(issue.RepoID) |
||||||
|
if err != nil { |
||||||
|
return fmt.Errorf("GetWatchers [%d]: %v", issue.RepoID, err) |
||||||
|
} |
||||||
|
|
||||||
|
tos := make([]string, 0, len(watchers)) // List of email addresses.
|
||||||
|
names := make([]string, 0, len(watchers)) |
||||||
|
for i := range watchers { |
||||||
|
if watchers[i].UserID == doer.Id { |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
to, err := GetUserByID(watchers[i].UserID) |
||||||
|
if err != nil { |
||||||
|
return fmt.Errorf("GetUserByID [%d]: %v", watchers[i].UserID, err) |
||||||
|
} |
||||||
|
if to.IsOrganization() { |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
tos = append(tos, to.Email) |
||||||
|
names = append(names, to.Name) |
||||||
|
} |
||||||
|
SendIssueCommentMail(issue, doer, tos) |
||||||
|
|
||||||
|
// Mail mentioned people and exclude watchers.
|
||||||
|
names = append(names, doer.Name) |
||||||
|
tos = make([]string, 0, len(mentions)) // list of user names.
|
||||||
|
for i := range mentions { |
||||||
|
if com.IsSliceContainsStr(names, mentions[i]) { |
||||||
|
continue |
||||||
|
} |
||||||
|
|
||||||
|
tos = append(tos, mentions[i]) |
||||||
|
} |
||||||
|
SendIssueMentionMail(issue, doer, GetUserEmailsByNames(tos)) |
||||||
|
|
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
// MailParticipants sends new issue thread created emails to repository watchers
|
||||||
|
// and mentioned people.
|
||||||
|
func (issue *Issue) MailParticipants() (err error) { |
||||||
|
mentions := markdown.FindAllMentions(issue.Content) |
||||||
|
if err = UpdateIssueMentions(issue.ID, mentions); err != nil { |
||||||
|
return fmt.Errorf("UpdateIssueMentions [%d]: %v", issue.ID, err) |
||||||
|
} |
||||||
|
|
||||||
|
if err = mailIssueCommentToParticipants(issue, issue.Poster, mentions); err != nil { |
||||||
|
log.Error(4, "mailIssueCommentToParticipants: %v", err) |
||||||
|
} |
||||||
|
|
||||||
|
return nil |
||||||
|
} |
@ -0,0 +1,183 @@ |
|||||||
|
// 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" |
||||||
|
"html/template" |
||||||
|
"path" |
||||||
|
|
||||||
|
"gopkg.in/gomail.v2" |
||||||
|
"gopkg.in/macaron.v1" |
||||||
|
|
||||||
|
"github.com/gogits/gogs/modules/base" |
||||||
|
"github.com/gogits/gogs/modules/log" |
||||||
|
"github.com/gogits/gogs/modules/mailer" |
||||||
|
"github.com/gogits/gogs/modules/markdown" |
||||||
|
"github.com/gogits/gogs/modules/setting" |
||||||
|
) |
||||||
|
|
||||||
|
const ( |
||||||
|
MAIL_AUTH_ACTIVATE base.TplName = "auth/activate" |
||||||
|
MAIL_AUTH_ACTIVATE_EMAIL base.TplName = "auth/activate_email" |
||||||
|
MAIL_AUTH_RESET_PASSWORD base.TplName = "auth/reset_passwd" |
||||||
|
MAIL_AUTH_REGISTER_NOTIFY base.TplName = "auth/register_notify" |
||||||
|
|
||||||
|
MAIL_ISSUE_COMMENT base.TplName = "issue/comment" |
||||||
|
MAIL_ISSUE_MENTION base.TplName = "issue/mention" |
||||||
|
|
||||||
|
MAIL_NOTIFY_COLLABORATOR base.TplName = "notify/collaborator" |
||||||
|
) |
||||||
|
|
||||||
|
type MailRender interface { |
||||||
|
HTMLString(string, interface{}, ...macaron.HTMLOptions) (string, error) |
||||||
|
} |
||||||
|
|
||||||
|
var mailRender MailRender |
||||||
|
|
||||||
|
func InitMailRender(dir, appendDir string, funcMap []template.FuncMap) { |
||||||
|
opt := &macaron.RenderOptions{ |
||||||
|
Directory: dir, |
||||||
|
AppendDirectories: []string{appendDir}, |
||||||
|
Funcs: funcMap, |
||||||
|
Extensions: []string{".tmpl", ".html"}, |
||||||
|
} |
||||||
|
ts := macaron.NewTemplateSet() |
||||||
|
ts.Set(macaron.DEFAULT_TPL_SET_NAME, opt) |
||||||
|
|
||||||
|
mailRender = &macaron.TplRender{ |
||||||
|
TemplateSet: ts, |
||||||
|
Opt: opt, |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func SendTestMail(email string) error { |
||||||
|
return gomail.Send(&mailer.Sender{}, mailer.NewMessage([]string{email}, "Gogs Test Email!", "Gogs Test Email!").Message) |
||||||
|
} |
||||||
|
|
||||||
|
func SendUserMail(c *macaron.Context, u *User, tpl base.TplName, code, subject, info string) { |
||||||
|
data := map[string]interface{}{ |
||||||
|
"Username": u.DisplayName(), |
||||||
|
"ActiveCodeLives": setting.Service.ActiveCodeLives / 60, |
||||||
|
"ResetPwdCodeLives": setting.Service.ResetPwdCodeLives / 60, |
||||||
|
"Code": code, |
||||||
|
} |
||||||
|
body, err := mailRender.HTMLString(string(tpl), data) |
||||||
|
if err != nil { |
||||||
|
log.Error(3, "HTMLString: %v", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
msg := mailer.NewMessage([]string{u.Email}, subject, body) |
||||||
|
msg.Info = fmt.Sprintf("UID: %d, %s", u.Id, info) |
||||||
|
|
||||||
|
mailer.SendAsync(msg) |
||||||
|
} |
||||||
|
|
||||||
|
func SendActivateAccountMail(c *macaron.Context, u *User) { |
||||||
|
SendUserMail(c, u, MAIL_AUTH_ACTIVATE, u.GenerateActivateCode(), c.Tr("mail.activate_account"), "activate account") |
||||||
|
} |
||||||
|
|
||||||
|
func SendResetPasswordMail(c *macaron.Context, u *User) { |
||||||
|
SendUserMail(c, u, MAIL_AUTH_RESET_PASSWORD, u.GenerateActivateCode(), c.Tr("mail.reset_password"), "reset password") |
||||||
|
} |
||||||
|
|
||||||
|
// SendActivateAccountMail sends confirmation email.
|
||||||
|
func SendActivateEmailMail(c *macaron.Context, u *User, email *EmailAddress) { |
||||||
|
data := map[string]interface{}{ |
||||||
|
"Username": u.DisplayName(), |
||||||
|
"ActiveCodeLives": setting.Service.ActiveCodeLives / 60, |
||||||
|
"Code": u.GenerateEmailActivateCode(email.Email), |
||||||
|
"Email": email.Email, |
||||||
|
} |
||||||
|
body, err := mailRender.HTMLString(string(MAIL_AUTH_ACTIVATE_EMAIL), data) |
||||||
|
if err != nil { |
||||||
|
log.Error(3, "HTMLString: %v", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
msg := mailer.NewMessage([]string{email.Email}, c.Tr("mail.activate_email"), body) |
||||||
|
msg.Info = fmt.Sprintf("UID: %d, activate email", u.Id) |
||||||
|
|
||||||
|
mailer.SendAsync(msg) |
||||||
|
} |
||||||
|
|
||||||
|
// SendRegisterNotifyMail triggers a notify e-mail by admin created a account.
|
||||||
|
func SendRegisterNotifyMail(c *macaron.Context, u *User) { |
||||||
|
data := map[string]interface{}{ |
||||||
|
"Username": u.DisplayName(), |
||||||
|
} |
||||||
|
body, err := mailRender.HTMLString(string(MAIL_AUTH_REGISTER_NOTIFY), data) |
||||||
|
if err != nil { |
||||||
|
log.Error(3, "HTMLString: %v", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
msg := mailer.NewMessage([]string{u.Email}, c.Tr("mail.register_notify"), body) |
||||||
|
msg.Info = fmt.Sprintf("UID: %d, registration notify", u.Id) |
||||||
|
|
||||||
|
mailer.SendAsync(msg) |
||||||
|
} |
||||||
|
|
||||||
|
// SendCollaboratorMail sends mail notification to new collaborator.
|
||||||
|
func SendCollaboratorMail(u, doer *User, repo *Repository) { |
||||||
|
repoName := path.Join(repo.Owner.Name, repo.Name) |
||||||
|
subject := fmt.Sprintf("%s added you to %s", doer.DisplayName(), repoName) |
||||||
|
|
||||||
|
data := map[string]interface{}{ |
||||||
|
"Subject": subject, |
||||||
|
"RepoName": repoName, |
||||||
|
"Link": repo.FullLink(), |
||||||
|
} |
||||||
|
body, err := mailRender.HTMLString(string(MAIL_NOTIFY_COLLABORATOR), data) |
||||||
|
if err != nil { |
||||||
|
log.Error(3, "HTMLString: %v", err) |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
msg := mailer.NewMessage([]string{u.Email}, subject, body) |
||||||
|
msg.Info = fmt.Sprintf("UID: %d, add collaborator", u.Id) |
||||||
|
|
||||||
|
mailer.SendAsync(msg) |
||||||
|
} |
||||||
|
|
||||||
|
func composeTplData(subject, body, link string) map[string]interface{} { |
||||||
|
data := make(map[string]interface{}, 10) |
||||||
|
data["Subject"] = subject |
||||||
|
data["Body"] = body |
||||||
|
data["Link"] = link |
||||||
|
return data |
||||||
|
} |
||||||
|
|
||||||
|
func composeIssueMessage(issue *Issue, doer *User, tplName base.TplName, tos []string, info string) *mailer.Message { |
||||||
|
subject := issue.MailSubject() |
||||||
|
body := string(markdown.RenderSpecialLink([]byte(issue.Content), issue.Repo.FullLink(), issue.Repo.ComposeMetas())) |
||||||
|
data := composeTplData(subject, body, issue.FullLink()) |
||||||
|
data["Doer"] = doer |
||||||
|
content, err := mailRender.HTMLString(string(tplName), data) |
||||||
|
if err != nil { |
||||||
|
log.Error(3, "HTMLString (%s): %v", tplName, err) |
||||||
|
} |
||||||
|
msg := mailer.NewMessage(tos, subject, content) |
||||||
|
msg.Info = fmt.Sprintf("Subject: %s, %s", subject, info) |
||||||
|
return msg |
||||||
|
} |
||||||
|
|
||||||
|
// SendIssueCommentMail composes and sends issue comment emails to target receivers.
|
||||||
|
func SendIssueCommentMail(issue *Issue, doer *User, tos []string) { |
||||||
|
if len(tos) == 0 { |
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
mailer.SendAsync(composeIssueMessage(issue, doer, MAIL_ISSUE_COMMENT, tos, "issue comment")) |
||||||
|
} |
||||||
|
|
||||||
|
// SendIssueMentionMail composes and sends issue mention emails to target receivers.
|
||||||
|
func SendIssueMentionMail(issue *Issue, doer *User, tos []string) { |
||||||
|
if len(tos) == 0 { |
||||||
|
return |
||||||
|
} |
||||||
|
mailer.SendAsync(composeIssueMessage(issue, doer, MAIL_ISSUE_MENTION, tos, "issue mention")) |
||||||
|
} |
@ -1,190 +0,0 @@ |
|||||||
// Copyright 2014 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 mailer |
|
||||||
|
|
||||||
import ( |
|
||||||
"fmt" |
|
||||||
"path" |
|
||||||
"strings" |
|
||||||
|
|
||||||
"gopkg.in/gomail.v2" |
|
||||||
"gopkg.in/macaron.v1" |
|
||||||
|
|
||||||
"github.com/gogits/gogs/models" |
|
||||||
"github.com/gogits/gogs/modules/base" |
|
||||||
"github.com/gogits/gogs/modules/log" |
|
||||||
"github.com/gogits/gogs/modules/markdown" |
|
||||||
"github.com/gogits/gogs/modules/setting" |
|
||||||
) |
|
||||||
|
|
||||||
const ( |
|
||||||
AUTH_ACTIVATE base.TplName = "mail/auth/activate" |
|
||||||
AUTH_ACTIVATE_EMAIL base.TplName = "mail/auth/activate_email" |
|
||||||
AUTH_REGISTER_NOTIFY base.TplName = "mail/auth/register_notify" |
|
||||||
AUTH_RESET_PASSWORD base.TplName = "mail/auth/reset_passwd" |
|
||||||
|
|
||||||
NOTIFY_COLLABORATOR base.TplName = "mail/notify/collaborator" |
|
||||||
NOTIFY_MENTION base.TplName = "mail/notify/mention" |
|
||||||
) |
|
||||||
|
|
||||||
func ComposeTplData(u *models.User) map[interface{}]interface{} { |
|
||||||
data := make(map[interface{}]interface{}, 10) |
|
||||||
data["AppName"] = setting.AppName |
|
||||||
data["AppVer"] = setting.AppVer |
|
||||||
data["AppUrl"] = setting.AppUrl |
|
||||||
data["ActiveCodeLives"] = setting.Service.ActiveCodeLives / 60 |
|
||||||
data["ResetPwdCodeLives"] = setting.Service.ResetPwdCodeLives / 60 |
|
||||||
|
|
||||||
if u != nil { |
|
||||||
data["User"] = u |
|
||||||
} |
|
||||||
return data |
|
||||||
} |
|
||||||
|
|
||||||
func SendUserMail(c *macaron.Context, u *models.User, tpl base.TplName, code, subject, info string) { |
|
||||||
data := ComposeTplData(u) |
|
||||||
data["Code"] = code |
|
||||||
body, err := c.HTMLString(string(tpl), data) |
|
||||||
if err != nil { |
|
||||||
log.Error(4, "HTMLString: %v", err) |
|
||||||
return |
|
||||||
} |
|
||||||
|
|
||||||
msg := NewMessage([]string{u.Email}, subject, body) |
|
||||||
msg.Info = fmt.Sprintf("UID: %d, %s", u.Id, info) |
|
||||||
|
|
||||||
SendAsync(msg) |
|
||||||
} |
|
||||||
|
|
||||||
func SendActivateAccountMail(c *macaron.Context, u *models.User) { |
|
||||||
SendUserMail(c, u, AUTH_ACTIVATE, u.GenerateActivateCode(), c.Tr("mail.activate_account"), "activate account") |
|
||||||
} |
|
||||||
|
|
||||||
// SendResetPasswordMail sends reset password e-mail.
|
|
||||||
func SendResetPasswordMail(c *macaron.Context, u *models.User) { |
|
||||||
SendUserMail(c, u, AUTH_RESET_PASSWORD, u.GenerateActivateCode(), c.Tr("mail.reset_password"), "reset password") |
|
||||||
} |
|
||||||
|
|
||||||
// SendRegisterNotifyMail triggers a notify e-mail by admin created a account.
|
|
||||||
func SendRegisterNotifyMail(c *macaron.Context, u *models.User) { |
|
||||||
body, err := c.HTMLString(string(AUTH_REGISTER_NOTIFY), ComposeTplData(u)) |
|
||||||
if err != nil { |
|
||||||
log.Error(4, "HTMLString: %v", err) |
|
||||||
return |
|
||||||
} |
|
||||||
|
|
||||||
msg := NewMessage([]string{u.Email}, c.Tr("mail.register_notify"), body) |
|
||||||
msg.Info = fmt.Sprintf("UID: %d, registration notify", u.Id) |
|
||||||
|
|
||||||
SendAsync(msg) |
|
||||||
} |
|
||||||
|
|
||||||
// SendActivateAccountMail sends confirmation e-mail.
|
|
||||||
func SendActivateEmailMail(c *macaron.Context, u *models.User, email *models.EmailAddress) { |
|
||||||
data := ComposeTplData(u) |
|
||||||
data["Code"] = u.GenerateEmailActivateCode(email.Email) |
|
||||||
data["Email"] = email.Email |
|
||||||
body, err := c.HTMLString(string(AUTH_ACTIVATE_EMAIL), data) |
|
||||||
if err != nil { |
|
||||||
log.Error(4, "HTMLString: %v", err) |
|
||||||
return |
|
||||||
} |
|
||||||
|
|
||||||
msg := NewMessage([]string{email.Email}, c.Tr("mail.activate_email"), body) |
|
||||||
msg.Info = fmt.Sprintf("UID: %d, activate email", u.Id) |
|
||||||
|
|
||||||
SendAsync(msg) |
|
||||||
} |
|
||||||
|
|
||||||
// SendIssueNotifyMail sends mail notification of all watchers of repository.
|
|
||||||
func SendIssueNotifyMail(u, owner *models.User, repo *models.Repository, issue *models.Issue) ([]string, error) { |
|
||||||
ws, err := models.GetWatchers(repo.ID) |
|
||||||
if err != nil { |
|
||||||
return nil, fmt.Errorf("GetWatchers[%d]: %v", repo.ID, err) |
|
||||||
} |
|
||||||
|
|
||||||
tos := make([]string, 0, len(ws)) |
|
||||||
for i := range ws { |
|
||||||
uid := ws[i].UserID |
|
||||||
if u.Id == uid { |
|
||||||
continue |
|
||||||
} |
|
||||||
to, err := models.GetUserByID(uid) |
|
||||||
if err != nil { |
|
||||||
return nil, fmt.Errorf("GetUserByID: %v", err) |
|
||||||
} |
|
||||||
if to.IsOrganization() { |
|
||||||
continue |
|
||||||
} |
|
||||||
|
|
||||||
tos = append(tos, to.Email) |
|
||||||
} |
|
||||||
|
|
||||||
if len(tos) == 0 { |
|
||||||
return tos, nil |
|
||||||
} |
|
||||||
|
|
||||||
subject := fmt.Sprintf("[%s] %s (#%d)", repo.Name, issue.Name, issue.Index) |
|
||||||
content := fmt.Sprintf("%s<br>-<br> <a href=\"%s%s/%s/issues/%d\">View it on Gogs</a>.", |
|
||||||
markdown.RenderSpecialLink([]byte(strings.Replace(issue.Content, "\n", "<br>", -1)), owner.Name+"/"+repo.Name, repo.ComposeMetas()), |
|
||||||
setting.AppUrl, owner.Name, repo.Name, issue.Index) |
|
||||||
msg := NewMessage(tos, subject, content) |
|
||||||
msg.Info = fmt.Sprintf("Subject: %s, issue notify", subject) |
|
||||||
|
|
||||||
SendAsync(msg) |
|
||||||
return tos, nil |
|
||||||
} |
|
||||||
|
|
||||||
// SendIssueMentionMail sends mail notification for who are mentioned in issue.
|
|
||||||
func SendIssueMentionMail(r macaron.Render, u, owner *models.User, |
|
||||||
repo *models.Repository, issue *models.Issue, tos []string) error { |
|
||||||
|
|
||||||
if len(tos) == 0 { |
|
||||||
return nil |
|
||||||
} |
|
||||||
|
|
||||||
subject := fmt.Sprintf("[%s] %s (#%d)", repo.Name, issue.Name, issue.Index) |
|
||||||
|
|
||||||
data := ComposeTplData(nil) |
|
||||||
data["IssueLink"] = fmt.Sprintf("%s/%s/issues/%d", owner.Name, repo.Name, issue.Index) |
|
||||||
data["Subject"] = subject |
|
||||||
data["ActUserName"] = u.DisplayName() |
|
||||||
data["Content"] = string(markdown.RenderSpecialLink([]byte(issue.Content), owner.Name+"/"+repo.Name, repo.ComposeMetas())) |
|
||||||
|
|
||||||
body, err := r.HTMLString(string(NOTIFY_MENTION), data) |
|
||||||
if err != nil { |
|
||||||
return fmt.Errorf("HTMLString: %v", err) |
|
||||||
} |
|
||||||
|
|
||||||
msg := NewMessage(tos, subject, body) |
|
||||||
msg.Info = fmt.Sprintf("Subject: %s, issue mention", subject) |
|
||||||
|
|
||||||
SendAsync(msg) |
|
||||||
return nil |
|
||||||
} |
|
||||||
|
|
||||||
// SendCollaboratorMail sends mail notification to new collaborator.
|
|
||||||
func SendCollaboratorMail(r macaron.Render, u, doer *models.User, repo *models.Repository) error { |
|
||||||
subject := fmt.Sprintf("%s added you to %s/%s", doer.Name, repo.Owner.Name, repo.Name) |
|
||||||
|
|
||||||
data := ComposeTplData(nil) |
|
||||||
data["RepoLink"] = path.Join(repo.Owner.Name, repo.Name) |
|
||||||
data["Subject"] = subject |
|
||||||
|
|
||||||
body, err := r.HTMLString(string(NOTIFY_COLLABORATOR), data) |
|
||||||
if err != nil { |
|
||||||
return fmt.Errorf("HTMLString: %v", err) |
|
||||||
} |
|
||||||
|
|
||||||
msg := NewMessage([]string{u.Email}, subject, body) |
|
||||||
msg.Info = fmt.Sprintf("UID: %d, add collaborator", u.Id) |
|
||||||
|
|
||||||
SendAsync(msg) |
|
||||||
return nil |
|
||||||
} |
|
||||||
|
|
||||||
func SendTestMail(email string) error { |
|
||||||
return gomail.Send(&Sender{}, NewMessage([]string{email}, "Gogs Test Email!", "Gogs Test Email!").Message) |
|
||||||
} |
|
@ -0,0 +1,17 @@ |
|||||||
|
<!DOCTYPE html> |
||||||
|
<html> |
||||||
|
<head> |
||||||
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> |
||||||
|
<title>{{.Subject}}</title> |
||||||
|
</head> |
||||||
|
|
||||||
|
<body> |
||||||
|
<p>@{{.Doer.Name}} mentioned you:</p> |
||||||
|
<p>{{.Body | Str2html}}</p> |
||||||
|
<p> |
||||||
|
--- |
||||||
|
<br> |
||||||
|
<a href="{{.Link}}">View it on Gogs</a>. |
||||||
|
</p> |
||||||
|
</body> |
||||||
|
</html> |
Loading…
Reference in new issue