是否有禁用所有电子邮件通知的漂亮解决方案?

Is there a beautiful solution for disable all email notification?

我有一个邮件服务,它只是向收件人发送通知。

@Service
public class MailService {

        @Autowired
        private JavaMailSender mailSender;

        public void send(String subject, String text, String... emails) {
                // MailMessage object configuration
                mailSender.send(mail);
        }
}

然后我有一个休息控制器,它在一些业务逻辑之后调用这个方法。

@RestController
public class MyRestController {

        @Autowired
        private MailService mailService;

        @PostMapping("/orders")
        public void createOrders(@RequestBody List<Order> orders) {
                // manipulations with orders
                mailService.send("Order notification", "New orders", "1@mail.com");
        }
}

我应该如何设计应用程序来关闭不同 spring 配置文件的通知?是否有根据配置文件配置电子邮件分发的最佳做法?

只需为需要它的配置文件提供虚拟实现

@Bean
@Profile("prod")
public JavaMailSender getRealJavaMailSender() {
    JavaMailSenderImpl mailSender = new JavaMailSenderImpl();
    mailSender.setHost("smtp.gmail.com");
    mailSender.setPort(587);
     
    mailSender.setUsername("my.gmail@gmail.com");
    mailSender.setPassword("password");
     
    Properties props = mailSender.getJavaMailProperties();
    props.put("mail.transport.protocol", "smtp");
    props.put("mail.smtp.auth", "true");
    props.put("mail.smtp.starttls.enable", "true");
    props.put("mail.debug", "true");
     
    return mailSender;
}

@Bean
@Profile("test")
public JavaMailSender getDummyJavaMailSender() {
    return new JavaMailSender() {
        ... dummy method implementations ...
    };
}

使用多个实现

一种可能性是创建一个具有两个实现的 MailService 接口。通过将这些实现与 @Profile 注释相结合,您可以根据您使用的配置文件正确地注入一个或另一个实现。例如:

public interface MailService {
    void send(String subject, String text, String... emails);
}

然后您可以将 @Profile("mail") 注释添加到一个实现中:

@Service
@Profile("mail") // Only create this bean when the 'mail' profile is used
public class JavaMailServiceImpl implements MailService {

    @Autowired
    private JavaMailSender mailSender;

    public void send(String subject, String text, String... emails) {
        // MailMessage object configuration
        mailSender.send(mail);
    }
}

此外,您可以在邮件配置文件未激活时添加另一个实现:

@Service
@Profile("!mail") // By adding the exclamination mark, this implementation will be used when the mail profile isn't active
public class NoopMailServiceImpl implements MailService {
    private final Logger logger = LoggerFactory.getLogger(getClass());

    @Override
    public void send(String subject, String text, String... emails) {
        logger.debug("Dummy implementation, no e-mail is being sent");
    }
}

如果您现在将 MailService 自动连接到您的控制器中,它将根据您使用的配置文件正确实施正确的实施。


使用方面

另一种方法是使用方面将发送邮件与订单创建逻辑完全分离。

这允许您从 MyRestController 控制器中完全删除 MailService 依赖项:

@RestController
public class MyRestController {

    @PostMapping("/orders")
    public void createOrders(@RequestBody List<Order> orders) {
        // manipulations with orders
    }
}

取而代之的是,您将 mailService 逻辑添加到一个单独的方面:

@Aspect
@Component
@Profile("mail")
public class OrderNotificationAspect {
    private final MailService mailService;

    public OrderNotificationAspect(MailService mailService) {
        this.mailService = mailService;
    }

    @AfterReturning("execution(* com.example.MyRestController.createOrders(..))")
    public void sendNotification() {
        mailService.send("Order notification", "New orders", "1@mail.com");
    }
}

和以前一样,我们使用 @Profile 注释仅在邮件配置文件处于活动状态时注册方面 bean。但是,由于我们不再直接将控制器绑定到 MailService,我们不再需要 "dummy implementation".

要在 Spring 引导中使用方面,您应该添加 spring-boot-starter-aop 依赖项。


以编程方式检查配置文件

另一种选择是在您的代码中编写检查,例如:

@RestController
public class MyRestController {

    @Autowired
    private MailService mailService;
    @Autowired
    private Environment environment;  

    @PostMapping("/orders")
    public void createOrders(@RequestBody List<Order> orders) {
        // manipulations with orders
        if (environment.acceptsProfiles(Profiles.of("mail"))) {
            mailService.send("Order notification", "New orders", "1@mail.com");
        }
    }
}

根据您希望此逻辑的细粒度程度,您可以将其放入控制器或服务中(如果您想禁用针对特定配置文件发送的所有电子邮件)。


您选择哪个选项取决于您的需要。使用多个实现是禁止发送所有邮件的最简单方法。

但是,如果您只想禁止发送某些邮件,则必须选择其他方式。使用方面的缺点是它有时会使流程更难以遵循,尽管 IDE 对方面有很好的支持。

我建议向 application-[profile-name].properties 文件添加一个变量(例如 sendEmails=false),每个配置文件具有不同的设置。

然后在您的代码中,您可以使用此 属性 来决定是否应发送电子邮件。

@Service
public class MailService {

    @Value("sendEmails}")
    private boolean sendEmails;

    @Autowired
    private JavaMailSender mailSender;

    public void send(String subject, String text, String... emails) {
        if (sendEmails) {
            // MailMessage object configuration
            mailSender.send(mail);
        }
    }
}

此方法使您能够为每个配置文件使用不同的(且易于更改的)设置。