Spring Boot利用Thymeleaf發(fā)送Email的方法教程
前言
眾所周知,現(xiàn)在在后臺(tái)服務(wù)器中發(fā)送郵件已經(jīng)是一個(gè)非常常用的功能了。通常來說雖然HTML并非是一個(gè)非常標(biāo)準(zhǔn)的信息格式,但是至少許多郵件客戶端都至少支持一部分標(biāo)記語言。 在這邊教程中主要是關(guān)于教你如何在Spring Boot 應(yīng)用中發(fā)送郵件以及使用非常簡單強(qiáng)大的Thymeleaf模板引擎來制作郵件內(nèi)容。
文章末尾附上源碼,已經(jīng)開源到Github上,是我公司做項(xiàng)目的時(shí)候處理郵件這一塊用到的。 基本上覆蓋了大部分郵件發(fā)送需求。稍微修改了一下,奉獻(xiàn)給有需要的人。當(dāng)你看完文章在看一下這封源碼,你會(huì)對(duì)這一塊更加的了解。而且你能掌握常用的郵件發(fā)送:
- 純文本郵件
- 內(nèi)聯(lián)圖片郵件
- 帶附件的郵件
純文本郵件
添加依賴(Mail starter dependencies)
首先制作并且通過SMTP郵件服務(wù)器來發(fā)送一個(gè)純文本郵件。
如果你之前有用過Spring Boot的話,那你寧該并不好奇在你建立一個(gè)新工程的時(shí)候,Spring Boot已經(jīng)幫你繼承了常用的依賴庫。 通常你只需要在你的 pom.xml 中添加如下依賴即可:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-mail</artifactId> </dependency>
郵件服務(wù)器屬性配置(Properties configuration)
通常情況下,如果所需要的依賴在 class path 中都是可用的話,這時(shí)候Spring會(huì)自動(dòng)幫你注冊一個(gè)默認(rèn)實(shí)現(xiàn)的郵件發(fā)送服務(wù) (default mail sender service)。 spring.mail.host 屬性已經(jīng)被自動(dòng)定義了, 所有我們所需要做的事情就是把這個(gè)屬性添加到我們應(yīng)用的 application.properties 配置文件中。
application.properties 在resource文件夾下
Spring Boot 提供的默認(rèn)郵件發(fā)送服務(wù) 其實(shí)已經(jīng)非常強(qiáng)大了,我們可以通過簡單的配置它的屬性就可以了。所謂的屬性其實(shí)說白了就是配置它的郵件SMTP 服務(wù)器:
spring.mail.port=25 # SMTP server port spring.mail.username= # Login used for authentication spring.mail.password= # Password for the given login spring.mail.protocol=smtp spring.mail.defaultEncoding=UTF-8 # Default message encoding
這里附帶一份 gmail 的SMTP服務(wù)器配置清單:
spring.mail.host = smtp.gmail.com spring.mail.username = *****@gmail.com spring.mail.password = **** spring.mail.properties.mail.smtp.auth = true spring.mail.properties.mail.smtp.socketFactory.port = 587 spring.mail.properties.mail.smtp.socketFactory.class = javax.net.ssl.SSLSocketFactory spring.mail.properties.mail.smtp.socketFactory.fallback = false
郵件發(fā)送服務(wù)(Mail sending service)
在這里我們使用 Autowired 在注入我們的service, 它主要就是生成郵件的相關(guān)信息
@Service public class MailClient { private JavaMailSender mailSender; @Autowired public MailService(JavaMailSender mailSender) { this.mailSender = mailSender; } public void prepareAndSend(String recipient, String message) { //TODO implement } }
生成郵件內(nèi)容
下面是一個(gè)簡單的生成郵件內(nèi)容的代碼。
public void prepareAndSend(String recipient, String message) { MimeMessagePreparator messagePreparator = mimeMessage -> { MimeMessageHelper messageHelper = new MimeMessageHelper(mimeMessage); messageHelper.setFrom("sample@dolszewski.com"); messageHelper.setTo(recipient); messageHelper.setSubject("Sample mail subject"); messageHelper.setText(message); }; try { mailSender.send(messagePreparator); } catch (MailException e) { // runtime exception; compiler will not force you to handle it } }
send() 需要被重寫以接受不同類型的參數(shù)變量:
- SimpleMailMessage: 正如名字所示,這是一個(gè)最基本的郵件message的模塊,我們可以給它設(shè)置常用的屬性,它并不能夠修改信息的頭,只能發(fā)送純文本的文件。
- MimeMessage: 通過這個(gè)類我們可以構(gòu)建出比較復(fù)雜的郵件內(nèi)容
- MimeMessagePreparator: 這是一個(gè)接口類,主要目的是提供一個(gè)構(gòu)建模板方法用來構(gòu)建 MimeMessage 以及當(dāng)你生成一個(gè)實(shí)例的時(shí)候幫你處理異常信息。官方文檔(也是常識(shí):))建議將MimeMessagePreparator作為郵件構(gòu)建的首選類型。
MimeMessageHelper類是MimeMessage的裝飾類,它提供了更多的開發(fā)人員友好界面,并為類的許多屬性添加了輸入驗(yàn)證。你可以不用,但是別人肯定會(huì)用,而且你會(huì)后悔不用 XD。
send() 會(huì)拋出 **MailException ** 異常,這是個(gè)運(yùn)行時(shí)異常,也就是通常所說的 RuntimeException。 在消息傳遞失敗的情況下,很可能會(huì)重復(fù)發(fā)送操作,或者至少使用一些更復(fù)雜的解決方案處理這種情況,例如:使用相應(yīng)的堆棧跟蹤記錄錯(cuò)誤消息。
手動(dòng)測試
通常如果你想郵件功能,你首先需要擁有一個(gè)SMTP服務(wù)器在你本機(jī)的電腦上處理你的請(qǐng)求。 如果你還沒用過,下面給你們推薦一些常用的:
- FakeSMTP – A simple server written in Java. Supported by any operating system with Java 1.6 or newer installed.
- smtp4dev – A server with a plain and user friendly interface. For Windows only.
- Papercut – Another simple server designed for Windows.
集成測試
你可能或許會(huì)感到好奇應(yīng)該如果寫一個(gè)自動(dòng)化的Test來驗(yàn)證你客戶端的功能。 如果你手動(dòng)測試的話,你需要開啟SMTP 服務(wù)器然后在運(yùn)行你的Spring Boot客戶端。 在這里給大家推薦一個(gè)神器 GreenMail, 因?yàn)樗鶭unit單元測試高度集成,可以簡化我們的測試。
添加依賴
GreenMail 已經(jīng)在Maven倉庫中了,所以我們唯一所需要做的就是將其依賴加入我們的 pom.xml 配置文件中:
<dependency> <groupId>com.icegreen</groupId> <artifactId>greenmail</artifactId> <version>1.5.0</version> <scope>test</scope> </dependency>
SMTP服務(wù)器與Test模板
現(xiàn)在呢,說了這么多廢話,我們終于可以創(chuàng)建我們的第一個(gè)集成測試類了。 它會(huì)啟動(dòng)Spring應(yīng)用程序并同時(shí)運(yùn)行郵件客戶端。但是在我們編寫實(shí)際測試之前呢,我們首先必須要確保SMTP服務(wù)器正確運(yùn)行,同時(shí)在測試結(jié)束的時(shí)候能夠正確關(guān)閉。
@RunWith(SpringJUnit4ClassRunner.class) @SpringApplicationConfiguration(Application.class) public class MailClientTest { private GreenMail smtpServer; @Before public void setUp() throws Exception { smtpServer = new GreenMail(new ServerSetup(25, null, "smtp")); smtpServer.start(); } @After public void tearDown() throws Exception { smtpServer.stop(); } }
創(chuàng)建郵件客戶端
首先,我們需要注入我們的郵件service在測試類中。之后,我們才能通過GrennMail來驗(yàn)證是否能夠接受到郵件。
@Autowired private MailClient mailClient; @Test public void shouldSendMail() throws Exception { //given String recipient = "name@hotmail.com"; String message = "Test message content"; //when mailClient.prepareAndSend(recipient, message); //then assertReceivedMessageContains(message); } private void assertReceivedMessageContains(String expected) throws IOException, MessagingException { MimeMessage[] receivedMessages = smtpServer.getReceivedMessages(); assertEquals(1, receivedMessages.length); String content = (String) receivedMessages[0].getContent(); assertTrue(content.contains(expected)); }
發(fā)送HTML郵件
在這里我們主要說一下如何構(gòu)建HTML類型的郵件。
Thymeleaf 模板引擎
首先在你的 pom.xml 中添加依賴。Spring引導(dǎo)將使用其默認(rèn)設(shè)置自動(dòng)準(zhǔn)備引擎
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-thymeleaf</artifactId> </dependency>
Thymeleaf的默認(rèn)配置期望所有HTML文件都放在 **resources/templates ** 目錄下,以.html擴(kuò)展名結(jié)尾。 讓我們創(chuàng)建一個(gè)名為mailTemplate.html的簡單文件,我們將使用創(chuàng)建的郵件客戶端類發(fā)送:
除了在生成過程中作為參數(shù)傳遞的消息的占位符,該模板幾乎不包含任何內(nèi)容。這不是廢話么-,-
模板處理
創(chuàng)建一個(gè)服務(wù)類,它主要負(fù)責(zé)將寫入的模板和外部模型組合在一起,這在我們的例子中是一個(gè)簡單的短信。
@Service public class MailContentBuilder { private TemplateEngine templateEngine; @Autowired public MailContentBuilder(TemplateEngine templateEngine) { this.templateEngine = templateEngine; } public String build(String message) { Context context = new Context(); context.setVariable("message", message); return templateEngine.process("mailTemplate", context); } }
注意:這里的 context。 這里主要使用的 鍵值對(duì) 的形式,類似map,將模板里面的需要的變量與值對(duì)應(yīng)起來。 比如: <span th:text="${message}"></span>
, 這里我們通過context就將message的內(nèi)容賦值給了span。
TemplateEngine類的實(shí)例由Spring Boot Thymeleaf自動(dòng)配置提供。我們所需要做的就是調(diào)用process()
方法,該方法接受兩個(gè)參數(shù),也就是我們使用的模板的名稱以及充當(dāng)模型的容器的上下文對(duì)象對(duì)象。
將新創(chuàng)建的 MailContentBuilder 注入到MailService類中。我們需要在prepareAndSen() 方法中進(jìn)行一個(gè)小的調(diào)整,以利用構(gòu)建器將生成內(nèi)容設(shè)置為mime消息。我們還使用 setText()
方法的重載變量將 Content-Type 頭設(shè)置為text / html,而不是默認(rèn)的 text / plain。
測試
需要更新的最后一件事是我們的測試,更確切地說,是接收到的消息的預(yù)期內(nèi)容。只需對(duì)驗(yàn)證邏輯進(jìn)行一個(gè)小的更改,運(yùn)行測試并檢查結(jié)果。
@Test public void shouldSendMail() throws Exception { //given String recipient = "name@dolszewski.com"; String message = "Test message content"; //when mailService.prepareAndSend(recipient, message); //then String content = "<span>" + message + "</span>"; assertReceivedMessageContains(content); }
本文到此基本宣告結(jié)束。
再次獻(xiàn)上一份我常用的html email模板:
<!DOCTYPE html> <html lang="en" xmlns="http://www.w3.org/1999/xhtml" xmlns:th="http://www.thymeleaf.org"> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/> <meta name="viewport" content="width=device-width, initial-scale=1.0"/> <title>Siemens Sinnovation</title> <style> .button { background-color: #4CAF50; border-radius: 12px; border: none; color: white; padding: 10px 25px; text-align: center; text-decoration: none; display: inline-block; font-size: 18px; margin: 4px 2px; cursor: pointer; box-shadow: 0 8px 16px 0 rgba(0, 0, 0, 0.2), 0 6px 20px 0 rgba(0, 0, 0, 0.19); } .button:hover { box-shadow: 0 12px 16px 0 rgba(0, 0, 0, 0.24), 0 17px 50px 0 rgba(0, 0, 0, 0.19); } </style> </head> <body style="margin: 0;padding: 0;"> <table align="center" border="1" cellpadding="0" cellspacing="0" width="600px"> <tr> <td> <table align="center" border="0" cellpadding="0" cellspacing="0" width="600" style="border-collapse: collapse;"> <tr> <td align="center" style="padding: 40px 0 30px 0;"> <!---->  </td> </tr> <tr> <td bgcolor="#ffffff" style="padding: 20px 30px 20px 30px"> <h4>The following message was created by <span th:text="${owner.getName()}"></span> in the Siemens DFFA group: </h4> <table border="0" cellpadding="0" cellspacing="0" width="100%"> <tr> <td><span th:text="${title}">title</span></td> </tr> <tr> <td style="padding: 20px 0 30px 0"> <span th:text="${description}">description</span> </td> </tr> <tr> <table border="0" cellpadding="0" cellspacing="0" width="100%"> <tr> <!--<td align="center" style="padding: 5px 0 3px 0">Status:<span--> <!--th:text="${status}">status</span></td>--> <!--<td align="center" style="padding: 5px 0 3px 0">Date submitted: <span--> <!--th:text="${createDate}">createDate</span></td>--> <!--<td align="center" style="padding: 5px 0 3px 0">Days left to join:<span--> <!--th:text="${leftTime}">leftTime</span></td>--> <td align="center" style="padding: 5px 0 3px 0">Status:<span th:text="${status}"> OPEN FOR JOINING</span></td> <td align="center" style="padding: 5px 0 3px 0">Date submitted: 28/08/2017 <span th:text="${createDate}">createDate</span></td> <td align="center" style="padding: 5px 0 3px 0">Days left to join: 10h<span th:text="${leftTime}">leftTime</span></td> </tr> </table> </tr> <tr> <table border="0" cellpadding="0" cellspacing="0" width="100%"> <tr> <td style="padding: 40px 0px 10px 0px"> Team Member: </td> </tr> <tr th:each="member :${members}"> <td style="padding: 5px 0px 5px 0px"><span th:text="${member.getName()}+', Email: '+${member.getEmail()}"></span> </td> </tr> </table> </tr> <tr> <td align="center" style="padding: 5px 40px 5px 40px"> <button class="button">View Details</button> </td> </tr> </table> </td> </tr> </table> </td> </tr> </table> </body> </html>
源碼下載:
Github 源碼:鏈接地址
本地下載:鏈接地址
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Java字符編碼簡介_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理
這篇文章主要介紹了Java字符編碼簡介,本文主要包括以下幾個(gè)方面:編碼基本知識(shí),Java,系統(tǒng)軟件,url,工具軟件等,感興趣的朋友一起看看吧2017-08-08深入淺出的學(xué)習(xí)Java ThreadLocal
本文會(huì)基于實(shí)際場景介紹ThreadLocal如何使用以及內(nèi)部實(shí)現(xiàn)機(jī)制。 具有很好的參考價(jià)值,下面跟著小編一起來看下吧2017-02-02SpringBoot使用AES對(duì)JSON數(shù)據(jù)加密和解密的實(shí)現(xiàn)方法
這篇文章主要介紹了SpringBoot使用AES對(duì)JSON數(shù)據(jù)加密和解密的實(shí)現(xiàn)方法,文章通過代碼示例介紹的非常詳細(xì),對(duì)我們的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-08-08你知道怎么用Spring的三級(jí)緩存解決循環(huán)依賴嗎
這篇文章主要為大家詳細(xì)介紹了Spring的三級(jí)緩存解決循環(huán)依賴,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-02-02springboot反爬蟲組件kk-anti-reptile的使用方法
這篇文章主要介紹了springboot反爬蟲組件kk-anti-reptile的使用方法,幫助大家更好的利用spring boot反爬蟲,保護(hù)網(wǎng)站安全,感興趣的朋友可以了解下2021-01-01使用Java實(shí)現(xiàn)KMZ和KML數(shù)據(jù)的直接解析
本文主要講解如何用JAVA語言,直接解析KMZ數(shù)據(jù),文章首先介紹google地圖中的KMZ和KML數(shù)據(jù),然后使用代碼的方式實(shí)現(xiàn)數(shù)據(jù)的解析,最后展示解析成果以及如何將數(shù)據(jù)轉(zhuǎn)換成空間WKT數(shù)據(jù),需要的朋友可以參考下2024-06-06