android - android中使用XOauth的Javamail api

标签 android oauth-2.0 jakarta-mail

我需要使用 javamail API 通过我的应用程序发送一封电子邮件(任何其他邮件服务,如果可用也可以)。问题是我不想问用户他的 usernamepassword

1) 是否可以将 OAuth 2.0JavaMail API/任何其他邮件 api

一起使用

2) 如何获取 OAuth Token ??

3) 网上有没有示例代码

提前致谢。

PS:我从未处理过邮件服务/SMTP 请求。

最佳答案

我对此进行了几天的研究,并找到了目前对我有用的解决方案。 我从 android AccountManager 获取 oauth2 token ,然后使用 JavaMail 通过 SMTP 发送电子邮件。这个想法是基于这里的 Java 示例 http://code.google.com/p/google-mail-oauth2-tools/wiki/JavaSampleCode在此 java Xoauth 示例中 http://google-mail-xoauth-tools.googlecode.com/svn/trunk/java/com/google/code/samples/xoauth/XoauthAuthenticator.java

JavaMail for Android 中没有有效的 SASL 实现,使用 asmack 也不起作用,所以我没有使用 SASL,而是像上面的 Xoauth 示例一样直接发出命令。

我像这样从账户经理那里得到 token

AccountManager am = AccountManager.get(this);
Account me = ...; //You need to get a google account on the device, it changes if you have more than one
am.getAuthToken(me, "oauth2:https://mail.google.com/", null, this, new OnTokenAcquired(), null);

private class OnTokenAcquired implements AccountManagerCallback<Bundle>{
    @Override
    public void run(AccountManagerFuture<Bundle> result){
        try{
            Bundle bundle = result.getResult();
            token = bundle.getString(AccountManager.KEY_AUTHTOKEN);

        } catch (Exception e){
            Log.d("test", e.getMessage());
        }
    }
}

如果有效,您的 token 中有一个 oauth2 token 。我在这段代码中使用了 token

import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.security.Provider;
import java.security.Security;
import java.util.Properties;

import javax.activation.DataHandler;
import javax.activation.DataSource;
import javax.mail.Session;
import javax.mail.Transport;
import javax.mail.URLName;
import javax.mail.Message;
import javax.mail.internet.InternetAddress;
import javax.mail.internet.MimeMessage;

import android.util.Log;

import com.sun.mail.smtp.SMTPTransport;
import com.sun.mail.util.BASE64EncoderStream;

public class GMailOauthSender {
private Session session;


public SMTPTransport connectToSmtp(String host, int port, String userEmail,
        String oauthToken, boolean debug) throws Exception {

    Properties props = new Properties();
    props.put("mail.smtp.starttls.enable", "true");
    props.put("mail.smtp.starttls.required", "true");
    props.put("mail.smtp.sasl.enable", "false");
    session = Session.getInstance(props);
    session.setDebug(debug);


    final URLName unusedUrlName = null;
    SMTPTransport transport = new SMTPTransport(session, unusedUrlName);
    // If the password is non-null, SMTP tries to do AUTH LOGIN.
    final String emptyPassword = null;
    transport.connect(host, port, userEmail, emptyPassword);

            byte[] response = String.format("user=%s\1auth=Bearer %s\1\1", userEmail,
            oauthToken).getBytes();
    response = BASE64EncoderStream.encode(response);

    transport.issueCommand("AUTH XOAUTH2 " + new String(response),
            235);

    return transport;
}

public synchronized void sendMail(String subject, String body, String user,
        String oauthToken, String recipients) {
    try {

        SMTPTransport smtpTransport = connectToSmtp("smtp.gmail.com",
                587,
                user,
                oauthToken,
                true);

        MimeMessage message = new MimeMessage(session);
        DataHandler handler = new DataHandler(new ByteArrayDataSource(body.getBytes(), "text/plain"));   
                message.setSender(new InternetAddress(user));   
                message.setSubject(subject);   
                message.setDataHandler(handler);   
        if (recipients.indexOf(',') > 0)   
            message.setRecipients(Message.RecipientType.TO, InternetAddress.parse(recipients));   
        else  
            message.setRecipient(Message.RecipientType.TO, new InternetAddress(recipients));   
        smtpTransport.sendMessage(message, message.getAllRecipients());   


    } catch (Exception e) {
        Log.d("test", e.getMessage());
    }

}

我根本不是这方面的专家,也没有使用上述示例中的任何安全提供程序,不确定它会如何影响这一点,但它对我有用。 希望这会有所帮助,并且有人可以告诉我这是否也有问题:p 这是我在这里的第一个答案,如果我做错了什么,请见谅!

Ops,忘记了我使用的其他一些文档:https://developers.google.com/google-apps/gmail/xoauth2_protocolhttp://developer.android.com/training/id-auth/authenticate.html

再次行动!您还需要 list 中的这些权限

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.USE_CREDENTIALS" />

关于android - android中使用XOauth的Javamail api,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12503303/

相关文章:

android - 在沉浸式 Activity 中添加 "ok glass contextual voice menu"

适用于 native 应用程序的 OAuth 2 - 公共(public)和 secret 客户端类型之间有什么区别?

android - 如何使用 Chrome 自定义选项卡进行 OAuth 2.0 登录 (Fitbit API)

java - 异常java后继续for循环

tomcat - javax.mail 不再适用于 java 11 和 tomcat 8.5

android - 查找 androidx 库版本的最佳方法是什么,例如 NavigationComponent、ViewModel 等?

android - 如何使用警报管理器进行通知?

android - Ti sensortag CC2650 噪声滤波器

戈朗 :exclude trash files from google drive

java - 无法使用 OAuth 2 通过 Jakarta Mail 连接到 Office 365 SMTP 服务器