1

很抱歉,英语不是我的母语,我的英语很烂。

我正在尝试使用带有 Python 的 Google Photos API 下载我的 3000 多张图片。

首先,我使用此代码下载了 MediaItems 列表。

from pathlib import Path
from requests_oauthlib import OAuth2Session
import json

api_url = "https://photoslibrary.googleapis.com/v1/mediaItems"
scope = ["https://www.googleapis.com/auth/photoslibrary.readonly"]


def save_token(token):
    token = {
        "access_token": token.get("access_token"),
        "refresh_token": token.get("refresh_token"),
        "token_type": token.get("token_type"),
        "expires_in": token.get("expires_in")
    }
    Path("token.json").write_text(json.dumps(token))


def load_token():
    token = {
        "access_token": "",
        "refresh_token": "",
        "token_type": "",
        "expires_in": "-30",
    }
    path = Path("token.json")
    if path.exists():
        token = json.loads(path.read_text())
    return token


def login():
    auth_info = json.loads(Path("credentials.json").read_text()).get("installed", None)
    assert auth_info is not None
    token = load_token()
    extras = {
        "client_id": auth_info.get("client_id"),
        "client_secret": auth_info.get("client_secret"),
    }
    google = OAuth2Session(
        auth_info.get("client_id"),
        scope=scope,
        token=token,
        auto_refresh_kwargs=extras,
        token_updater=save_token,
        auto_refresh_url=auth_info.get("token_uri"),
        redirect_uri=auth_info.get("redirect_uris")[0]
    )
    if not google.authorized:
        authorization_url, state = google.authorization_url(
            auth_info.get("auth_uri"),
            access_type="offline",
            prompt="select_account"
        )
        print("Access {} and paste code.".format(authorization_url))
        access_code = input(">>> ")
        google.fetch_token(
            auth_info.get("token_uri"),
            client_secret=auth_info.get("client_secret"),
            code=access_code
        )
        assert google.authorized
        save_token(google.token)
    return google


def test():
    google = login()
    response = google.get(api_url)
    print(response.text)


if __name__ == "__main__":
    test()

这段代码没有问题,我用 nextPageToken 下载了大约 30 个 json 文件(包含 3000 张图片信息)。

之后,我尝试通过此代码下载这些图片。

photo_info_list 变量包含所有媒体项。

photo_download_format = "{base}=w{width}-h{height}"

def download_photos(photo_info_list):
    google = login()
    for photo_info in photo_info_list:
        photo_id = photo_info.get("id", "dummy_id")
        base_url = photo_info.get("baseUrl")
        metadata = photo_info.get("mediaMetadata")
        filename = photo_info.get("filename")
        download_url = photo_download_format.format(
            base=base_url,
            width=metadata["width"],
            height=metadata["height"]
        )
        response = google.get(download_url)
        # save_picture

此代码适用于前 162 张图片(约 270MB),但随后出现 403 禁止错误。

我删除了令牌并再次尝试登录程序,创建了另一个凭据但得到了同样的错误。

有谁知道是什么问题?

任何建议或信息都非常感谢。谢谢!

4

1 回答 1

4

baseUrls 在获取后 60 分钟内过期。根据Google Photos API的文档

您也不应该存储 baseUrls,它会在大约 60 分钟后过期。

最可能的解释是您的 baseUrl 在下载过程中已过期。

于 2019-02-13T15:22:12.070 回答