Как автоматически обновлять учетные данные Gmail после истечения срока их действия

avatar
Dhanu K
8 апреля 2018 в 07:54
361
1
1

Я использую Gmail Java API, API работает нормально, и когда мы предоставляем учетные данные для доступа к учетной записи Google, они сохраняются локально, но через 60 минут срок действия учетных данных истекает, он не обновляет учетные данные автоматически. При отладке с помощью моего кода он останавливается на

Учетные данные = app.authorize("user");

Как справиться с истечением срока действия учетных данных, он снова работает только после вызова аутентификации oauth2. Возможно ли автоматическое обновление этой функции Это мой блок кода

/**
     * Creates an authorized Credential object.
     * @return an authorized Credential object.
     * @throws IOException
     */
    public static Credential authorize() throws IOException {
        // Load client secrets.
        InputStream in =
                Quickstart.class.getResourceAsStream("/client_secret.json");
        GoogleClientSecrets clientSecrets =
                GoogleClientSecrets.load(JSON_FACTORY, new InputStreamReader(in));
        HttpServletRequest request = ((ServletRequestAttributes)RequestContextHolder.getRequestAttributes()).getRequest();
        HttpSession session = request.getSession();
        User user =  (User) session.getAttribute("USER");
        java.io.File DATA_STORE_DIR = new java.io.File(
                //System.getProperty("user.home"), ".credentials/"+user.getUserAccountId()+"/gmail-java-quickstart");
                Common.commonPath+File.separator+"client_secrets"+File.separator+user.getUserAccountId()+"/gmail-java-quickstart");
        DATA_STORE_FACTORY = new FileDataStoreFactory(DATA_STORE_DIR);
        // Build flow and trigger user authorization request.
        GoogleAuthorizationCodeFlow flow =
                new GoogleAuthorizationCodeFlow.Builder(
                        HTTP_TRANSPORT, JSON_FACTORY, clientSecrets, SCOPES)
                        .setDataStoreFactory(DATA_STORE_FACTORY)
                        .setAccessType("offline")
                        .build();
        System.out.println("*************getApprovalPrompt:"+flow.getApprovalPrompt());
        AuthorizationCodeInstalledApp app = new AuthorizationCodeInstalledApp(flow, new LocalServerReceiver()){
            protected void onAuthorization(AuthorizationCodeRequestUrl authorizationUrl) throws java.io.IOException {
                //System.out.println("authorizationUrl:"+authorizationUrl);
            }
        };
        Credential credential = app.authorize("user");
        //System.out.println("*************AUTH_KKEY:"+credential.getAccessToken());
        //System.out.println("Credentials saved to " + DATA_STORE_DIR.getAbsolutePath());
        return credential;
    }

    /**
     * Build and return an authorized Gmail client service.
     * @return an authorized Gmail client service
     * @throws IOException
     */
    public static Gmail getGmailService() throws IOException {
        Credential credential = authorize();
        return new Gmail.Builder(HTTP_TRANSPORT, JSON_FACTORY, credential)
                .setApplicationName(APPLICATION_NAME)
                .build();
    }

    public List<Message> listMessagesWithLabels(List<String> labelIds,StringBuilder paginationtokenBuilder,Long maxResults) throws IOException {
        List<Message> messages = new ArrayList<Message>();
        try {
            Gmail service = getGmailService();
            // some logic bla bla
        }catch(Exception ex){
            ex.printStackTrace();
            return messages;
        }
    }
Источник
Emre Dalkiran
22 апреля 2018 в 16:37
0

Это немного поздно, но запускаете ли вы этот код более чем для одного адреса электронной почты для одной и той же учетной записи пользователя на одном компьютере? Ваш код кажется, что он должен работать нормально. Я могу думать только о том, что токен обновления не работает, потому что вы жестко закодировали userId, который равен "user". Это может быть перезапись учетных данных для разных пользователей одним и тем же userId.

Ответы (1)

avatar
Mohan Kanakala
10 апреля 2018 в 22:18
1

Эй, я столкнулся с той же проблемой, но не повезло с API. У меня есть кое-какая работа над проблемой, проверьте, поможет ли это.

По истечении срока действия токена API выдает исключение (или тайм-аут в некоторых случаях). Здесь токен может быть обновлен вручную (может быть с помощью блока catch). последующий запрос будет продолжен с обновленным токеном, который обновляется предыдущим запросом. Фрагмент кода ниже.

В случае исключения:

try {
    response1 = Quickstart.getGmailService().users().messages().list(userId).setLabelIds(labelIds).execute();       
    return response1.getResultSizeEstimate();
} catch (com.google.api.client.auth.oauth2.TokenResponseException e) {
    e.printStackTrace();
    FileUtils.deleteDirectory(new File(CLIENT_SECRET_PATH+File.separator+"client_secrets"));
    response.sendRedirect("/dashboard"); // redirect to your oauth request URI
} catch (IOException e) {
    e.printStackTrace();
}

В случае тайм-аута (UI):

$.ajax({
    url:"listGmailLabels", // Request that is hitting Gmail API
    type:"POST",
    success:function(data){
        // Success logic goes here
    },
    complete:function(event){
        if(event.status == 504){
            alert('Token expired! Getting a new refresh token.');
            $.ajax({
                url:"clearClientSecret", // service call to cleanup the expired token
                success:function(){
                    window.location.reload();
                }
            })
        }
    }
 }); 

Служба очистки токена с истекшим сроком действия:

@RequestMapping(value="/clearClientSecret",method= {RequestMethod.POST,RequestMethod.GET})
    public void clearClientSecret(HttpServletRequest request,HttpServletResponse response) {
        try {
            String path=CLIENT_SECRET_PATH+File.separator+"client_secrets";
            FileUtils.deleteDirectory(new File(path));
            response.sendRedirect("/dashboard");// redirect to your oauth request URI
        } catch (IOException e1) {
            e1.printStackTrace();
        }

    }

Я знаю, что это может быть неправильным решением, но это то, что я пробовал в качестве обходного пути, поэтому дайте нам всем знать, если есть что-то, что может улучшить его.