无法使用JavaMail通过SMTP发送SSL或TLS邮件

5

新年快乐!

我正在开发一个应用程序,当特定触发器发生时,用户将收到一封电子邮件。

这是我用来发送电子邮件的功能:

public static void sendEmail(String host, String port, String useSSL, String useTLS, String useAuth, String user, String password, String subject, String content, String type, String recipients)
            throws NoSuchProviderException, AddressException, MessagingException  {
        final Properties props = new Properties();
        props.setProperty("mail.transport.protocol", "smtp");
        props.setProperty("mail.smtp.host", host);
        props.setProperty("mail.smtp.port", port);        
        if (useSSL != null && !useSSL.equals("false") && useSSL.equals("true")) {
            props.setProperty("mail.smtp.ssl.enable", useSSL);
            props.setProperty("mail.smtp.socketFactory.class",
                    "javax.net.ssl.SSLSocketFactory");
            props.setProperty("mail.smtp.socketFactory.port", port);

        }
        if (useTLS != null && !useTLS.equals("false") && useTLS.equals("true")) {
            props.setProperty("mail.smtp.starttls.enable", useTLS);
            props.setProperty("mail.smtp.socketFactory.fallback", "true");
        }   
        props.setProperty("mail.smtp.auth", useAuth);
        props.setProperty("mail.from", user);  
        props.setProperty("mail.smtp.user", user);
        props.setProperty("mail.password", password);

        Session mailSession = Session.getDefaultInstance(props, new Authenticator() {
            protected PasswordAuthentication getPasswordAuthentication() {
                return new PasswordAuthentication(props.getProperty("mail.smtp.user"), props
                        .getProperty("mail.password"));
            }
        });   

        Transport transport = mailSession.getTransport();

        MimeMessage message = new MimeMessage(mailSession);
        message.setHeader("Subject", subject);
        message.setContent(content, type);

        StringTokenizer tokenizer = new StringTokenizer(recipients, ";");
        while (tokenizer.hasMoreTokens()) {
            String recipient = tokenizer.nextToken();
            message.addRecipient(Message.RecipientType.TO,
                    new InternetAddress(recipient));
        }

        transport.connect();
        transport.sendMessage(message, message.getRecipients(Message.RecipientType.TO));
        transport.close();

奇怪的是,每当我尝试用main方法运行上述代码时,它都可以成功地发送SSL和TLS协议的电子邮件。

public static void main(String args[])
    {
        try {
            Notifier.sendEmail("smtp.gmail.com", "587", "false", "true", "true","sender_email@gmail.com", "testpassword", "CHECKING SETTINGS", "CHECKING EMAIL FUNCTIONALITY", "text/html", "cc_email@gmail.com");
        } catch (Exception ex) {
            ex.printStackTrace();
        } 
    }

但是,每当我尝试通过我的Web应用程序运行相同的代码时,它就会失败。

通过SSL发送会引发此错误:

com.sun.mail.smtp.SMTPSendFailedException: 530-5.5.1 Authentication Required. Learn more at
jvm 1    | 530 5.5.1  https://support.google.com/mail/answer/14257 f12sm88286300pat.20 - gsmtp
jvm 1    | 
jvm 1    |  at com.sun.mail.smtp.SMTPTransport.issueSendCommand(SMTPTransport.java:2057)

通过TLS发送会出现以下错误:
javax.mail.MessagingException: Could not connect to SMTP host: smtp.gmail.com, port: 587;
jvm 1    |   nested exception is:
jvm 1    |  javax.net.ssl.SSLException: Unrecognized SSL message, plaintext connection?
jvm 1    |  at com.sun.mail.smtp.SMTPTransport.openServer(SMTPTransport.java:1934)

任何形式的帮助都会受到欢迎。
编辑1:
这是来自前端的tpl文件。
<div class="label1"><h3 class="label">Host:</h3></div>
     <div class="field1"><input type="text" class="input1" name="host" size="20" value="$HOST$"></div>
         <div class="port"><h3 class="label">Port:</h3></div>
         <div class="fieldport"><input type="text" class="fieldport" name="port" size="5" value="$PORT$"></div>
         <div class="ssl">
                <input type="radio" name="sslEnable" value="$SSLENABLE$">
                    Enable SSL?
         </div>
         <div class="tls">
                <input type="radio" name="tlsEnable" value="$TLSENABLE$">
                    Enable TLS?
         </div>
         <div class="auth">
                <input type="checkbox" name="auth"$AUTH$>
                    Enable Authentication?
         </div>            
     <div class="label2"><h3 class="label">User:</h3></div>
     <div class="field2"><input type="text" class="input1" name="user" size="20" value="$USER$"></div>
     <div class="label3"><h3 class="label">Password:</h3></div>
     <div class="field3"><input type="password" class="input1" name="password" size="20" value="$PASSWORD$"></div>
     <div class="label4"><h3 class="label">Recipient(s):</h3></div>
     <div class="field4"><input type="text" class="input1" name="recipients" size="50" value="$RECIPIENTS$"></div>

值会保存在类似于以下格式的配置文件中:

host=smtp.gmail.com
port=587
ssl=false
tls=true
auth=true
user=send_user_email@gmail.com
password=O0UbYboDfVFRaiA=
recipients=cc_user_email@gmail.com
trigger1=false
attempt=0
trigger2=false
percent=5
anyOrAll=ANY
trigger3=true
format=HTML
trigger4=true
trigger5=true

编辑2:

public static void sendEmail(String message)
      throws NoSuchProviderException, AddressException, MessagingException
  {
    if (message == null || message.trim().equals("")) return;

    StringBuffer content = new StringBuffer();
    content.append(getHeader());
    content.append(message);
    content.append(getFooter());
    String format = NotifyProps.getFormat();
    String type = "text/plain";
    if (format.equals(NotifyProps.HTML)) type = "text/html";

    sendEmail(NotifyProps.getHost(), NotifyProps.getPort(), Boolean.toString(NotifyProps.getUseAuth()), Boolean.toString(NotifyProps.getUseSSL()), Boolean.toString(NotifyProps.getUseTLS()),NotifyProps.getUser(), NotifyProps.getPassword(),
              "Transaction Processor Auto Notification", content.toString(), type,
              NotifyProps.getRecipients())
  }

这是一个设置和获取属性的类: https://codeshare.io/5G8ki 谢谢。

让我惊奇的是你说它在命令行测试中工作正常。即使是TLS,你也应该将mail.smtp.socketFactory.class设置为javax.net.ssl.SSLSocketFactory,而不仅仅是SSL。 但与此无关,我建议您使用Yoda条件来检查字符串常量,如"true".equals(sslStr)。它是空安全的,更简洁。 - coladict
5个回答

2
尽管您的代码看起来没问题,但至少存在一个大问题。您试图在TLS和SSL之间使用相同的端口(587)。我不确定TLS,但如果您将请求发送到端口465,则SSL代码应该正常工作。如此处(谷歌FAQ)所述:

配置SMTP服务器时,端口465(使用SSL)和端口587(使用TLS)各自有特定的端口号[...]

它们有自己的特定端口。SSL出现的错误:

Unrecognized SSL message, plaintext connection?

您的客户是否无法理解它收到了未经SSL编码的响应 (由于TLS端口未实现SSL导致)。


我没有同时使用相同的端口来处理这两种协议。 我使用465端口来处理SSL,使用587端口来处理TLS。 - Ravi.
你能展示一下实际产生错误的网页代码吗?你已经包含了主方法版本,但那已经可以工作了。根据错误信息,你正在尝试在TLS端口上初始化SSL连接。可能是你的Web应用程序参数列表出了问题? - Gergely Bacso
那个配置是如何被读取的?前端代码并不是很重要,重要的是加载这个配置的部分。这些配置条目(比如端口)最终会到达 sendMail 函数。看一下它们是否被正确地接收会很有帮助。 - Gergely Bacso
你上次链接使用的网站返回HTTP 522错误,所以我无法检查。能否记录下你的Web应用程序中的参数,以显示它获取的确切值?可能会读取冲突的参数,或返回默认值、缓存结果...各种错误都可能导致传递错误的端口。 - Gergely Bacso

1
我们为TLS设置了更多的属性。
props.put("mail.smtp.starttls.enable", "true");
props.setProperty("mail.smtp.ssl.enable", "true");
props.setProperty("mail.smtp.socketFactory.class", "javax.net.ssl.SSLSocketFactory");
props.put("mail.smtp.socketFactory.fallback", "false");

对于身份验证,请使用smtps而不是smtp

props.setProperty("mail.smtps.auth", useAuth);

在获取传输时
session.getTransport("smtps"); 

在连接时再次传递主机电子邮件和密码

transport.connect("smtp.gmail.com", "user@email", "password");

调试用


session.setDebug(true);

0

最近,Gmail进行了安全更新。您必须在https://myaccount.google.com/security?pli=1页面中允许“允许较不安全的应用程序访问”选项。然后,您就可以毫无问题地从您的帐户发送邮件了。


0

尝试以下代码,它对我有效。

public void sendEmail(){

        Properties props = new Properties();
        props.put("mail.smtp.host", "smtp.gmail.com");
        props.put("mail.smtp.socketFactory.port", "465");
        props.put("mail.smtp.socketFactory.class", "javax.net.ssl.SSLSocketFactory");
        props.put("mail.smtp.auth", "true");
        props.put("mail.smtp.port", "465");

Session session = Session.getDefaultInstance(props,
            new javax.mail.Authenticator() {
                protected PasswordAuthentication getPasswordAuthentication() {
                    return new PasswordAuthentication("senderEmail@gmail.com","secret");
                }
            });

            try {
            Message message = new MimeMessage(session);
            message.setFrom(new InternetAddress("from-email@gmail.com"));
            message.setRecipients(Message.RecipientType.TO,
                    InternetAddress.parse("to-email@gmail.com"));
            message.setSubject("This is testing message");
            message.setText("Hi this is testing email....not spam");

        Transport.send(message);
            System.out.println("email successfully sent..");

        } catch (MessagingException e) {
            throw new RuntimeException(e);
        }
    }

0

simple-java-mail提供了一个简单的枚举,可以用来指示 SSL 或 TLS。这样,您就不必担心正确的属性了:

Email email = new Email();

(...)

new Mailer("smtp.gmail.com", 25, "your user", "your password", TransportStrategy.SMTP_TLS).sendMail(email);
new Mailer("smtp.gmail.com", 587, "your user", "your password", TransportStrategy.SMTP_TLS).sendMail(email);
new Mailer("smtp.gmail.com", 465, "your user", "your password", TransportStrategy.SMTP_SSL).sendMail(email);

如果您已经开启了双重认证登录,您需要从您的Google账户生成一个应用程序专用密码才能使此示例工作。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接