使用SpringMail写一个告警发送邮件系统

没有bug的系统不是好系统!在我们日常的开发中系统报错了只会把错误信息打印在日志中,然后等着用户来反馈,程序员在着手处理,查日志,定位错误的代码行。

其实整个查看错误的过程是有好几步的,如果日志信息多了,那么排查错误就比较困难。那我们能不能程序发生错误了就马上通知程序员和运维人员呢?

答案是肯定能的,现在市面上有很多第三方类似的告警系统,甚至不止这一个发送邮件来告警,还有很多种方式。

但是我不想用第三方软件来做这么一个功能,我喜欢自己写一个,嘿嘿,程序员嘛,自己动手丰衣足食,足够满足自己的业务需求就好了。

为什么要使用发送邮件而不是发送验证码呢?因为发送验证码要钱,成本太高,发送邮件来告知是最方便也是成本最低的方案了。

好了,咱们开始动手写代码吧,先说说整个架构用的技术和版本,项目使用springBoot+springAop+springMail+SpringListener来搭建。

咱们使用的是smtp.163.com服务来发送邮件,所以咱们得有一个163邮箱账号,然后在设置页把POP3/SMTP/IMAP服务给开启。


配置好之后,咱们来开始写代码吧,我先把整个pom给贴出来。

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.1.6.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.spring.mail</groupId>
    <artifactId>demo</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>springMail</name>
    <description>Demo project for Spring Boot</description>

    <properties>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <!--引入aopjar-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>

        <dependency>
            <groupId>com.google.guava</groupId>
            <artifactId>guava</artifactId>
            <version>20.0</version>
        </dependency>

        <!--引入springMail jar包-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-mail</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-devtools</artifactId>
            <scope>runtime</scope>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-configuration-processor</artifactId>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>
</project>

主要的引入就是aop和springMail这两个dependency。

咱们再定义一个发送的模板消息实体类。

package com.spring.mail.bean;

import lombok.Data;
import lombok.ToString;

/**
 * 定义发送信息模板
 */
@Data
@ToString
public class MailWarningBean {

    /**
     * 错误信息
     */
    private String errorMsg;

    /**
     * 参数列表
     */
    private String params;

    /**
     * 类路径
     */
    private String path;
}

再把yml配置文件的代码贴出来

server:
  port: 8080
  tomcat:
    uri-encoding: utf-8
spring:
  aop:
    auto: true
  mail:
    host: 220.181.12.11   #linux有时候不能有效的进行域名解析,所以我们直接用smtp.163.com来ping一下获取真实的ip地址
    username: XXXXXX  #填写你的邮箱地址(发送人)
    password: XXXXXX  #你的邮箱密码
    default-encoding: UTF-8
    properties:
      mail:
        smtp:
          auth: true
          port: 465  #使用465端口 因为在服务器上25端口不能用
          socketFactory:
            port: 465
            class: javax.net.ssl.SSLSocketFactory
            fallback: false
          starttls:
            enable: true
            required: true
    port: 465

咱们的思路主要是:使用spring的事件监听来定义邮件的发送,使用aop的异常增强来发送邮件。当异常产生了就发送邮件,告知咱们的程序员。

先定义spring的事件监听模块。


package com.spring.mail.event;

import lombok.Data;
import org.springframework.context.ApplicationEvent;

/**
 * 定义事件
 * @param <T>
 */
@Data
public class MyEvent<T> extends ApplicationEvent {

    private T t;

    public MyEvent(Object source,T t){
        super(source);
        this.t=t;
    }
}

事件类已经写好,咱们就开始动手写事件的监听类,主要就是做邮件的发送,sendMail方法使用了Async注解,Async注解就是开启异步,让发送邮件异步来处理。

package com.spring.mail.event.listener;

import com.google.common.base.Throwables;
import com.spring.mail.bean.MailWarningBean;
import com.spring.mail.event.MyEvent;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.event.EventListener;
import org.springframework.mail.javamail.JavaMailSender;
import org.springframework.mail.javamail.MimeMessageHelper;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Component;

import javax.mail.internet.MimeMessage;

@Component
@Slf4j
public class MailListener {

    @Autowired
    private JavaMailSender javaMailSender;


    /**
     * 定义事件、开启异步
     * @param myEvent
     */
    @EventListener
    @Async
    public void sendMail(MyEvent<MailWarningBean> myEvent){
        MimeMessage mimeMessage=null;
        try{
            MailWarningBean mailWarningBean=myEvent.getT();
            mimeMessage=javaMailSender.createMimeMessage();
            MimeMessageHelper msgHelper=new MimeMessageHelper(mimeMessage,true);
            //发送人邮箱
            msgHelper.setFrom("XXXXXXX");
            //收件人邮箱(单个)
            msgHelper.setTo("XXXXXXXX");
            /**
             * 发送多人
             * //收件人
             *             InternetAddress[] internetAddresses=new InternetAddress[]{
             *                     new InternetAddress("XXXXXXX.com","","utf-8"),
             *                     new InternetAddress("XXXXXXX.com","","utf-8")
             *             };
             *             msgHelper.setTo(internetAddresses);
             */
            //主题
            msgHelper.setSubject("服务器系统传错误,请尽快处理!");
            //拼接好发送的内容
            StringBuffer sb=new StringBuffer();
            sb.append("<p><h2>您的类方法路径"+mailWarningBean.getPath()+"出现错误!</h2></p>");
            sb.append("<p>参数列表:"+mailWarningBean.getParams()+"</p>");
            sb.append("<p>错误信息:<p style='color:red;'>"+mailWarningBean.getErrorMsg()+"。</p>");
            sb.append("请尽快处理。</p><p>出现错误服务器ip地址:<strong>XXX.XXX.XX</strong></p>");
            msgHelper.setText(sb.toString(),true);
            javaMailSender.send(mimeMessage);   //进行发送
            log.info("发送邮件成功");
        }catch(Exception e){
            log.error("发送告警邮件出现错误,错误原因:"+ Throwables.getStackTraceAsString(e));
        }
    }
}

咱们再来做aop的处理,aop的增强方式有前置增强、后置增强、环绕增强、异常增强等方式,咱们采用异常增强,因为咱们主要的业务就是做告警,当程序出错了咱们就发送邮件告知运维人员和开发人员。来,我把aop的代码贴出来。

package com.spring.mail.aop;

import com.google.common.base.Throwables;
import com.spring.mail.bean.MailWarningBean;
import com.spring.mail.event.MyEvent;
import com.spring.mail.event.listener.MailListener;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

@Aspect
@Component
public class MailAspect {

    @Autowired
    private MailListener mailListener;

    //定义切面
    @Pointcut("execution(public * com.spring.mail.controller.*.*(..))")
    public void MailAspect(){}


    /**
     * 使用aop的异常增强方式来发送邮件
     * @param joinPoint
     * @param e
     */
    @AfterThrowing(pointcut = "MailAspect()",throwing = "e")
    public void deAfterThrowing(JoinPoint joinPoint,Throwable e){
        Object[] param=joinPoint.getArgs(); //获取参数列表
        StringBuilder sb=new StringBuilder();
        for(Object obj:param){
           sb.append(obj+",");
        }
        MailWarningBean mailWarningBean=new MailWarningBean();
        mailWarningBean.setErrorMsg(Throwables.getStackTraceAsString(e));
        mailWarningBean.setParams(sb.delete(0,sb.length()).toString());
        mailWarningBean.setPath(joinPoint.getSignature().getDeclaringTypeName()+"."+joinPoint.getSignature().getName()); //获取方法路径
        MyEvent<MailWarningBean> myEvent=new MyEvent<>(this,mailWarningBean);
        mailListener.sendMail(myEvent);
    }
}

整个配置就写的差不多了,在aop类上使用@Pointcut来定义切面,当controller出现异常时发送邮件。接下来,咱们就测试测试吧。

先写一个controller

package com.spring.mail.controller;

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class MainController {


    @GetMapping("/hello")
    public String hello(){
        System.out.println(1/0);
        return "test";
    }
}

在接口内故意写一行代码产生异常,看会不会接收到邮件。



访问接口时报错了,日志信息打印出了邮件已发送,咱们现在看看邮箱有没有收到告警的邮件。


可以看到邮箱是收到了此邮件,说明咱们整个告警模块系统就写好了。完整的代码可以点下面的链接去git克隆代码,有不懂的地方欢迎大家留言提问和讨论。

git仓库:https://github.com/wuyanzu01/springMail

请关注微信公众号:请快点喜欢我

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,393评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,790评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,391评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,703评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,613评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,003评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,507评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,158评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,300评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,256评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,274评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,984评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,569评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,662评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,899评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,268评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,840评论 2 339

推荐阅读更多精彩内容