解决Salesforce无法向外发送邮件

问题

有时,我们想在Salesforce组织(Org)中自动对外发送邮件。例如,我们可以使用下面的Apex代码实现。

首先,创建一个方便我们快速发送邮件至单个收件人地址的类。

java 复制代码
// 创建一个方便复用,负责发送单个邮件的功能类
public class EmailManager {
	
	// 主要的发送邮件方法
    public void sendMail(String address, String subject, String body) {
        // 创建一个新邮件消息对象
        Messaging.SingleEmailMessage mail = new Messaging.SingleEmailMessage();

        // 设置邮件消息的收件地址、主题和正文内容
        mail.setToAddresses(new String[] { address }); // 尽管目标地址只有一个,但因Salesforce默认设计为方便批量发送,必须以列表(数组)形式传参
        mail.setSubject(subject);
        mail.setPlainTextBody(body);
        // 因默认方便批量发送,`发送结果`对象也必须用列表(数组)捕捉
        Messaging.SendEmailResult[] results = Messaging.sendEmail(
            new Messaging.SingleEmailMessage[] {mail}
        );
        inspectResults(results);
    }

    // 一个仅供内部使用的私有方法,用于快速检查邮件是否全部发送成功
    private static Boolean inspectResults(Messaging.SendEmailResult[] results) {
        // 默认所有邮件都发送成功
        Boolean sendResult = true;
        
        for (Messaging.SendEmailResult res : results) {
            // 对于每个发送结果,检查是否成功
            if (res.isSuccess()) {
                System.debug('Email sent successfully');
            }
            else {
                // 如果失败,将`所有邮件都发送成功`设置为否,并在运行日志中提示
                sendResult = false;
                System.debug('The following errors occurred: ' + res.getErrors());
            }
        }
        
        return sendResult;
    }
}

之后,调用我们自己编写的类------创建一个对象,并用其发送邮件。

java 复制代码
EmailManager em = new EmailManager();
em.sendMail(
    'someone@gmail.com',
    'Test Email',
    'This is a test body'
);

现象

然而,过了很久,我们的Gmail邮箱中都并无邮件显示,哪怕是垃圾邮件分类中。

故障排查

检查日志

这一现象非常反常。我们首先应检查日志,检查是否发送成功。

log 复制代码
66.0 APEX_CODE,FINEST;APEX_PROFILING,INFO;CALLOUT,INFO;DATA_ACCESS,INFO;DB,INFO;NBA,INFO;SYSTEM,DEBUG;VALIDATION,INFO;VISUALFORCE,INFO;WAVE,INFO;WORKFLOW,INFO
Execute Anonymous: EmailManager em = new EmailManager();
Execute Anonymous: em.sendMail(
Execute Anonymous:     'someone@gmail.com',
Execute Anonymous:     'Test Emaill',
Execute Anonymous:     'This is a test body'
Execute Anonymous: );
19:08:22.1 (1715616)|USER_INFO|[EXTERNAL]|...

然而,我们程序打印消息,声称发送成功。

log 复制代码
...
19:08:22.1 (85500107)|USER_DEBUG|[18]|DEBUG|Email sent successfully
...

而且,日志也显示邮件被成功加入邮件发送队列。

log 复制代码
19:08:22.1 (83919155)|EMAIL_QUEUE|[8]|subject: Test Email, bccSender: false, saveAsActivity: true, useSignature: true, toAddresses: [someone@gmail.com], plainTextBody: This is a test body, 
...

这十分诡异。让我们检查一下组织(Org)设置。

检查可交付性设置

每个组织中都有一处可以设置可交付性Deliverability)------控制是否允许邮件发出。

然而,当前设置已经允许向任何邮件发送出站(outbound)邮件。

这十分反常。为了进一步测试,我们直接用系统的测试功能向邮箱发送测试邮件。

然而我们却未曾收到任何内容。

检查邮箱端

也许这是Gmail的问题?让我们更换到Hotmail (Outlook) 试试。将目标邮箱更改为一Outlook邮箱,再次发送测试邮件。

我们立刻收到了邮件,尽管位于垃圾邮件中。

既然如此,让我们重新运行匿名Apex代码,发送调用之前编写的类邮件。

java 复制代码
EmailManager em = new EmailManager();
em.sendMail(
    'someone@hotmail.com',
    'Test Email',
    'This is a test body'
);

我们立刻收到了Apex代码发送的邮件。问题解决。

结论

有时,Gmail的垃圾邮件屏蔽策略过于激进,反而导致了不必要的问题。问题可能出在接收端,而非发送端(Apex)。

相关推荐
plainGeekDev4 分钟前
Activity 间传值 → Navigation 参数
android·java·kotlin
用户41659673693554 分钟前
Android WebView 加载 file:// 离线页面调试教程
android·前端
plainGeekDev8 分钟前
onActivityResult → ActivityResult API
android·java·kotlin
Sunia9 分钟前
《AgentX 专栏》10-生产部署:3台2C4G云服务器把企业级Agent真正跑起来的完整方案
java·架构
ZhengEnCi1 小时前
J7A-高级Java工程师面试三道灵魂拷问-深度广度与工程素养的终极检验
java·后端
随遇丿而安4 小时前
第10周:Activity 基础功能与生命周期优化
android
alexhilton17 小时前
Android车载OS中的Remote Compose
android·kotlin·android jetpack
狼爷19 小时前
吃透 Java Function 接口,搞定 99% 的 Stream 场景
java·函数式编程
祎雪双十Gy1 天前
从 DataX 的配置加载说起:我用 FastJson2 做了一个轻量级动态配置管理库
java·后端
小锋java12341 天前
分享一套锋哥原创的SpringBoot4+Vue3宠物领养网站系统
java