Szczęśliwego Nowego Roku!Nie można wysyłać poczty za pośrednictwem protokołu SSL lub TLS za pomocą protokołu SMTP przy użyciu Javamail
Pracuję nad aplikacją, w której użytkownik otrzymuje wiadomość e-mail, gdy wystąpi określony problem.
Jest to funkcja, która używam do wysyłania e-mail:
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();
dość dziwne, gdy próbuję uruchomić powyższy kod z metody Main że to wysłać e-mail dla obu protokołów SSL i TLS z powodzeniem.
public static void main(String args[])
{
try {
Notifier.sendEmail("smtp.gmail.com", "587", "false", "true", "true","[email protected]", "testpassword", "CHECKING SETTINGS", "CHECKING EMAIL FUNCTIONALITY", "text/html", "[email protected]");
} catch (Exception ex) {
ex.printStackTrace();
}
}
Ale zawiedzie, gdy próbuję uruchomić ten sam kod za pośrednictwem mojej aplikacji internetowej.
wysyłanie go poprzez SSL generuje ten błąd:
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)
wysłanie go poprzez TLS generuje ten błąd:
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)
Wszelkiego rodzaju pomoc jest mile widziana.
Edit1:
Oto plik tpl z przodu
<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>
Wartości zostaje zapisany w pliku konfiguracyjnym, który wygląda tak:
host=smtp.gmail.com
port=587
ssl=false
tls=true
auth=true
[email protected]
password=O0UbYboDfVFRaiA=
[email protected]
trigger1=false
attempt=0
trigger2=false
percent=5
anyOrAll=ANY
trigger3=true
format=HTML
trigger4=true
trigger5=true
Edit2:
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())
}
Jest to klasa, który wyznacza i pobiera właściwości:
Dziękuję.
To, co zaskakuje mnie, to fakt, że działa w teście wiersza poleceń. Powinieneś ustawić 'mail.smtp.socketFactory.class' na' javax.net.ssl.SSLSocketFactory' nawet z TLS, nie tylko SSL. Ale niezwiązane z tym, proponuję użyć warunków Yoda, aby sprawdzić stałe ciąg znaków, takich jak '" true ".equals (sslStr)'. Jest bezpieczny i ma mniej bałaganu. – coladict