6

我必须实现一个将文件上传到 Google Drive 的功能。
快速入门指南中所述。

首先,我调用 generateLinkAuth 生成链接来获取代码。
然后将执行 generateCode 以获取令牌。
一切都很好。

    List<String> SCOPES = Arrays.asList("https://www.googleapis.com/auth/drive.file", "https://www.googleapis.com/auth/userinfo.email",
            "https://www.googleapis.com/auth/userinfo.profile", DriveScopes.DRIVE);
    public String generateLinkAuth() {
            mHttpTransport = new NetHttpTransport();
            mJsonFactory = new JacksonFactory();

            mFlow = new GoogleAuthorizationCodeFlow.Builder(mHttpTransport, mJsonFactory, CLIENT_ID, CLIENT_SECRET, SCOPES).setAccessType("online").setApprovalPrompt("force").build();


            String url = mFlow.newAuthorizationUrl().setRedirectUri(REDIRECT_URI).build();

            System.out.println("url :" + url);
            return url;
        }

        public void generateCode(String code) {
            new GetCredentialFromCode(mContext, code).execute();
        }


        class GetCredentialFromCode extends AsyncTask<Void, Void, Void> {
            private Context mContext;
            private String code;
            private final ProgressDialog mDialog;

            public GetCredentialFromCode(Context context, String cd) {
                mContext = context;
                mDialog = new ProgressDialog(context);
                mDialog.show();
                mDialog.setTitle("Uploading ...");
                code = cd;
            }

            @Override
            protected Void doInBackground(Void... params) {

                if (mFlow != null) {
                    try {

                        mResponse = mFlow.newTokenRequest(code).setRedirectUri(REDIRECT_URI).execute();

                        mCredential = new GoogleCredential();

                        mCredential.setFromTokenResponse(mResponse);


                        String accessToken = mCredential.getAccessToken();
                        Editor edit = mPrefs.edit();
                        edit.putString(GOOGLE_DRIVE_TOKEN, accessToken);
                        edit.commit();

                        String accessTokenSave = mPrefs.getString(GOOGLE_DRIVE_TOKEN, "NO KEY");
                        Log.e(TAG, "access Token Saved is " + accessTokenSave);

                    } catch (IOException e) {
                        Editor edit = mPrefs.edit();
                        edit.putString(GOOGLE_DRIVE_TOKEN, "NO KEY");
                        e.printStackTrace();
                    }
                }

                String accessTokenSave = mPrefs.getString(GOOGLE_DRIVE_TOKEN, "NO KEY");

                mHttpTransport = new NetHttpTransport();
                mJsonFactory = new JacksonFactory();
                mCredential = new GoogleCredential().setAccessToken(accessTokenSave);

                getUserInfo();

                mDriveService = new Drive.Builder(mHttpTransport, mJsonFactory, mCredential).build();
                return null;
            }

            @Override
            protected void onPostExecute(Void result) {
                mDialog.dismiss();
                GoogleDriveLogInAct act=  (GoogleDriveLogInAct) mContext;
                act.goBackSettingScreen();
            }

        }

问题是令牌将在 1 小时内使用 setAccessType("online") 过期,所以我将其更改为 setAccessType("offline")。
当我更改它并再次运行我的 android 应用程序时,出现错误 - 当我执行以下代码时。

                mResponse = mFlow.newTokenRequest(code).setRedirectUri(REDIRECT_URI).execute();

                mCredential = new GoogleCredential();

                mCredential.setFromTokenResponse(mResponse);

当我尝试在控制台上调试时,执行mCredential.setFromTokenResponse(mResponse)。, 将引发异常

java.lang.RuntimeException: An error occured while executing doInBackground()
    at android.os.AsyncTask$3.done(AsyncTask.java:278)
    at java.util.concurrent.FutureTask$Sync.innerSetException(FutureTask.java:273)
    at java.util.concurrent.FutureTask.setException(FutureTask.java:124)
    at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:307)
    at java.util.concurrent.FutureTask.run(FutureTask.java:137)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1076)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:569)
    at java.lang.Thread.run(Thread.java:856)
 Caused by: java.lang.IllegalArgumentException: Please use the Builder and call setJsonFactory, setTransport and setClientSecrets
    at com.google.common.base.Preconditions.checkArgument(Preconditions.java:88)
    at com.google.api.client.googleapis.auth.oauth2.GoogleCredential.setRefreshToken(GoogleCredential.java:294)
    at com.google.api.client.googleapis.auth.oauth2.GoogleCredential.setRefreshToken(GoogleCredential.java:153)
    at com.google.api.client.auth.oauth2.Credential.setFromTokenResponse(Credential.java:556)
    at com.google.api.client.googleapis.auth.oauth2.GoogleCredential.setFromTokenResponse(GoogleCredential.java:313)
    at com.jp.shotdocs.libs.GoogleDriveLib$GetCredentialFromCode.doInBackground(GoogleDriveLib.java:168)
    at com.jp.shotdocs.libs.GoogleDriveLib$GetCredentialFromCode.doInBackground(GoogleDriveLib.java:1)
    at android.os.AsyncTask$2.call(AsyncTask.java:264)
    at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
    ... 4 more

如何离线使用 accessType?
有什么我必须修改才能使用这种模式吗?

4

1 回答 1

1

(Answered in the comments. See Question with no answers, but issue solved in the comments (or extended in chat) )

The OP wrote:

have solve this problem... I should declare GoogleCredential by using GoogleCredential.Build() instead of new GoogleCredential().

于 2015-01-26T16:09:55.767 回答