17

javamail APIたとえば、 (利用可能な場合は他のメールサービスでも可能です)を使用して、アプリからメールを送信する必要があります。username問題は、ユーザーに彼とを尋ねたくないということですpassword

1)/他のメールAPIOAuth 2.0と一緒に使用することは可能ですか?JavaMail API

2)OAuthトークンを取得する方法??

3)ネット上で利用可能なサンプルコードはありますか

前もって感謝します。

PS:私はメールサービス/SMTPリクエストを扱ったことがありません。

4

1 に答える 1

28

私はこれを数日間調査し、現在私のために働いている解決策を見つけました。AndroidのAccountManagerからoauth2トークンを取得し、JavaMailを使用してSMTP経由でメールを送信します。このアイデアは、Javaの例http://code.google.com/p/google-mail-oauth2-tools/wiki/JavaSampleCodeと、このJavaXoauthの例http://google-mail-xoauth-toolsに基づいています。 .googlecode.com / svn / trunk / java / com / google / code / samples / xoauth / XoauthAuthenticator.java

JavaMail for Androidには動作するSASL実装がなく、asmackの使用も機能していなかったため、SASLを使用せず、上記のXoauthの例のようにコマンドを直接発行しました。

このようにアカウントマネージャーからトークンを取得します

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());
        }
    }
}

それが機能する場合は、トークンにoauth2トークンがあります。このコードでトークンを使用します

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_protocolおよびhttp://developer.android.com/training/id-auth/authenticate.html

もう一度おっと!マニフェストにもこれらの権限が必要です

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.USE_CREDENTIALS" />
于 2012-10-10T14:19:14.147 に答える