| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | // Copyright 2022 The Gitea Authors. All rights reserved. | 
					
						
							| 
									
										
										
										
											2022-11-27 13:20:29 -05:00
										 |  |  | // SPDX-License-Identifier: MIT | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | package templates | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"html/template" | 
					
						
							| 
									
										
										
										
											2023-04-30 20:22:23 +08:00
										 |  |  | 	"regexp" | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 	"strings" | 
					
						
							|  |  |  | 	texttmpl "text/template" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-08 21:15:22 +08:00
										 |  |  | 	"code.gitea.io/gitea/modules/base" | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 	"code.gitea.io/gitea/modules/log" | 
					
						
							|  |  |  | 	"code.gitea.io/gitea/modules/setting" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-30 20:22:23 +08:00
										 |  |  | var mailSubjectSplit = regexp.MustCompile(`(?m)^-{3,}\s*$`) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-08 21:15:22 +08:00
										 |  |  | // mailSubjectTextFuncMap returns functions for injecting to text templates, it's only used for mail subject | 
					
						
							|  |  |  | func mailSubjectTextFuncMap() texttmpl.FuncMap { | 
					
						
							|  |  |  | 	return texttmpl.FuncMap{ | 
					
						
							|  |  |  | 		"dict": dict, | 
					
						
							|  |  |  | 		"Eval": Eval, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		"EllipsisString": base.EllipsisString, | 
					
						
							|  |  |  | 		"AppName": func() string { | 
					
						
							|  |  |  | 			return setting.AppName | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 		"AppDomain": func() string { // documented in mail-templates.md | 
					
						
							|  |  |  | 			return setting.Domain | 
					
						
							|  |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func buildSubjectBodyTemplate(stpl *texttmpl.Template, btpl *template.Template, name string, content []byte) { | 
					
						
							|  |  |  | 	// Split template into subject and body | 
					
						
							|  |  |  | 	var subjectContent []byte | 
					
						
							|  |  |  | 	bodyContent := content | 
					
						
							|  |  |  | 	loc := mailSubjectSplit.FindIndex(content) | 
					
						
							|  |  |  | 	if loc != nil { | 
					
						
							|  |  |  | 		subjectContent = content[0:loc[0]] | 
					
						
							|  |  |  | 		bodyContent = content[loc[1]:] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if _, err := stpl.New(name). | 
					
						
							|  |  |  | 		Parse(string(subjectContent)); err != nil { | 
					
						
							|  |  |  | 		log.Warn("Failed to parse template [%s/subject]: %v", name, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if _, err := btpl.New(name). | 
					
						
							|  |  |  | 		Parse(string(bodyContent)); err != nil { | 
					
						
							|  |  |  | 		log.Warn("Failed to parse template [%s/body]: %v", name, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | // Mailer provides the templates required for sending notification mails. | 
					
						
							|  |  |  | func Mailer(ctx context.Context) (*texttmpl.Template, *template.Template) { | 
					
						
							| 
									
										
										
										
											2023-04-08 21:15:22 +08:00
										 |  |  | 	subjectTemplates := texttmpl.New("") | 
					
						
							|  |  |  | 	bodyTemplates := template.New("") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	subjectTemplates.Funcs(mailSubjectTextFuncMap()) | 
					
						
							| 
									
										
										
										
											2023-04-30 20:22:23 +08:00
										 |  |  | 	bodyTemplates.Funcs(NewFuncMap()) | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 18:16:45 +08:00
										 |  |  | 	assetFS := AssetFS() | 
					
						
							| 
									
										
										
										
											2023-05-25 11:47:30 +08:00
										 |  |  | 	refreshTemplates := func(firstRun bool) { | 
					
						
							|  |  |  | 		if !firstRun { | 
					
						
							|  |  |  | 			log.Trace("Reloading mail templates") | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-04-12 18:16:45 +08:00
										 |  |  | 		assetPaths, err := ListMailTemplateAssetNames(assetFS) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			log.Error("Failed to list mail templates: %v", err) | 
					
						
							|  |  |  | 			return | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 18:16:45 +08:00
										 |  |  | 		for _, assetPath := range assetPaths { | 
					
						
							|  |  |  | 			content, layerName, err := assetFS.ReadLayeredFile(assetPath) | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2023-04-12 18:16:45 +08:00
										 |  |  | 				log.Warn("Failed to read mail template %s by %s: %v", assetPath, layerName, err) | 
					
						
							|  |  |  | 				continue | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2023-04-12 18:16:45 +08:00
										 |  |  | 			tmplName := strings.TrimPrefix(strings.TrimSuffix(assetPath, ".tmpl"), "mail/") | 
					
						
							| 
									
										
										
										
											2023-05-25 11:47:30 +08:00
										 |  |  | 			if firstRun { | 
					
						
							|  |  |  | 				log.Trace("Adding mail template %s: %s by %s", tmplName, assetPath, layerName) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2023-04-12 18:16:45 +08:00
										 |  |  | 			buildSubjectBodyTemplate(subjectTemplates, bodyTemplates, tmplName, content) | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-25 11:47:30 +08:00
										 |  |  | 	refreshTemplates(true) | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if !setting.IsProd { | 
					
						
							|  |  |  | 		// Now subjectTemplates and bodyTemplates are both synchronized | 
					
						
							|  |  |  | 		// thus it is safe to call refresh from a different goroutine | 
					
						
							| 
									
										
										
										
											2023-05-25 11:47:30 +08:00
										 |  |  | 		go assetFS.WatchLocalChanges(ctx, func() { | 
					
						
							|  |  |  | 			refreshTemplates(false) | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2022-08-28 10:43:25 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return subjectTemplates, bodyTemplates | 
					
						
							|  |  |  | } |