AccessTokenRefreshError:oauth2client的invalid_grant

5 投票
1 回答
787 浏览
提问于 2025-04-18 15:38

我正在创建一个脚本,目的是为特定的Google Apps用户下载文件,使用的是Google的Python客户端库。首先,我想获取用户账户中包含的文件列表。

我正在参考这个链接中的示例:https://developers.google.com/drive/v2/reference/files/list

下面是代码块:

用户授权客户端

def authorize_application(request):

    #setting flow to get permission and code 
    flow = OAuth2WebServerFlow(CLIENT_ID, CLIENT_SECRET, OAUTH_SCOPE, REDIRECT_URI, ACCESS_TYPE)
    authorize_url = flow.step1_get_authorize_url()

    code = request.GET.get('code', '')

    if code:
        #import ipdb
        #ipdb.set_trace()
    #setting flow step2 to exchage code for access token    
        credential = flow.step2_exchange(code)

    #initialising httplib2 instance and building a DriveAPI service
        http = httplib2.Http()
        http = credential.authorize(http)
        drive_service = build('drive', 'v2', http=http)

        # getting user's data
    about = drive_service.about().get().execute()
        user_info = about.get('user')
        email = user_info.get('emailAddress')
        request.session['username'] = email[0:email.index('@')]
        username = request.session['username']
        #import ipdb
        #ipdb.set_trace()
        #creating a Django user object 
    user, created = User.objects.get_or_create(username=username, email=email)

    #saving credentials to database 
        if created == True: 
            storage = Storage(CredentialsModel, 'id', user, 'credential')
            storage.put(credential)
            return HttpResponseRedirect('/download/')
        else:
        return HttpResponseRedirect('/download/')
    else:
        return HttpResponseRedirect(authorize_url)

获取用户Google Drive中的文件列表

def download_file(request):

    #import ipdb
    #ipdb.set_trace()

    username = request.session['username']
    user = User.objects.get(username=username)

    storage = Storage(CredentialsModel, 'id', user, 'credential')
    credential = storage.get()

    access_token = credential.access_token

    http = httplib2.Http()
    http = credential.authorize(http)
    drive_service = build('drive', 'v2', http=http)

    result = []
    page_token = access_token

    while True:

    try:
        param = {}
            if page_token:
                param['pageToken'] = page_token
            files = drive_service.files().list(**param).execute()

            result.extend(files['items'])
                page_token = files.get('nextPageToken')

        if not page_token:
        break
    except errors.HttpError, error:
        print 'An error occured: %s' % error
        break

    return HttpResponse(result)

这个脚本能够从数据库中保存和获取凭证,但在执行下载文件的代码时,出现了错误:“AccessTokenRefreshError: invalid_grant”,具体是在这行代码:files = drive_service.files().list(**param).execute()。

我哪里做错了?请给我一些正确的建议。

1 个回答

1

(2017年2月) 不太确定你遇到的问题是什么,但有几点需要注意:1)Drive API有一个更新版本(v3),2)你使用的认证代码现在可以通过最近更新的Google API客户端库大大简化。(确保你用 pip install -U google-api-python-client [或者用 pip3 适用于Python 3] 更新你的Python库。)

下面是一个有效的解决方案——告诉我它是否对你有效——我在这篇博客这个视频中写的。我省略了 about() 调用来获取用户信息,但我把代码分成了你上面提到的两个部分,不过可以看到,代码量少了很多:

用户对客户端的授权

SCOPES = 'https://www.googleapis.com/auth/drive.readonly.metadata'
store = file.Storage('storage.json')
creds = store.get()
if not creds or creds.invalid:
    flow = client.flow_from_clientsecrets('client_id.json', SCOPES)
    creds = tools.run_flow(flow, store)

获取用户Drive中的文件列表(前100个文件)

DRIVE = discovery.build('drive', 'v3', http=creds.authorize(Http()))
files = DRIVE.files().list().execute().get('files', [])
for f in files:
    print(f['name'], f['mimeType'])

如果你想了解更多关于使用Drive API的内容,这里有一些我创建的额外资源(视频、博客文章等):

(*) - 简而言之:将纯文本文件上传到Drive,导入/转换为Google文档格式,然后将该文档导出为PDF。上面的文章使用的是Drive API v2,和你的代码示例一样;这篇后续文章描述了如何迁移到Drive API v3,还有一个开发者视频结合了两个“穷人的转换器”文章。

撰写回答