web-dev-qa-db-ja.com

ステータスが例外を発生させる400のようなエラーである場合、Python urllibの応答本文を読み取る方法は?

Python 3 urllibを使用してGitHubAPIにリクエストを送信してリリースを作成しようとしていますが、間違いがあり、例外が発生して失敗します。

Traceback (most recent call last):
  File "./a.py", line 27, in <module>
    'Authorization': 'token ' + token,
  File "/usr/lib/python3.6/urllib/request.py", line 223, in urlopen
    return opener.open(url, data, timeout)
  File "/usr/lib/python3.6/urllib/request.py", line 532, in open
    response = meth(req, response)
  File "/usr/lib/python3.6/urllib/request.py", line 642, in http_response
    'http', request, response, code, msg, hdrs)
  File "/usr/lib/python3.6/urllib/request.py", line 570, in error
    return self._call_chain(*args)
  File "/usr/lib/python3.6/urllib/request.py", line 504, in _call_chain
    result = func(*args)
  File "/usr/lib/python3.6/urllib/request.py", line 650, in http_error_default
    raise HTTPError(req.full_url, code, msg, hdrs, fp)
urllib.error.HTTPError: HTTP Error 422: Unprocessable Entity

ただし、GitHubは優れており、次のように応答本文で失敗した理由を説明しています。 データのPOST に対する400対422の応答

では、どのように応答本文を読み取るのですか?例外が発生するのを防ぐ方法はありますか?

例外をキャッチしてipdbで調べようとしました。これにより、タイプ urllib.error.HTTPError しかし、そこにはその本文データは見つかりませんでした。ヘッダーのみが見つかりました。

スクリプト:

#!/usr/bin/env python3

import json
import os
import sys

from urllib.parse import urlencode
from urllib.request import Request, urlopen

repo = sys.argv[1]
tag = sys.argv[2]
upload_file = sys.argv[3]

token = os.environ['GITHUB_TOKEN']
url_template = 'https://{}.github.com/repos/' + repo + '/releases'

# Create.
_json = json.loads(urlopen(Request(
    url_template.format('api'),
    json.dumps({
        'tag_namezxcvxzcv': tag,
        'name': tag,
        'prerelease': True,
    }).encode(),
    headers={
        'Accept': 'application/vnd.github.v3+json',
        'Authorization': 'token ' + token,
    },
)).read().decode())
# This is not the tag, but rather some database integer identifier.
release_id = _json['id']

使用法: 誰かがpython githubにリリースアセットをアップロードする例をリクエストできますか?

HTTPErrorにはread()メソッドがあり、応答の本文を読み取ることができます。したがって、あなたの場合、次のようなことができるはずです。

try:
    body = urlopen(Request(
        url_template.format('api'),
        json.dumps({
            'tag_namezxcvxzcv': tag,
            'name': tag,
            'prerelease': True,
        }).encode(),
        headers={
            'Accept': 'application/vnd.github.v3+json',
            'Authorization': 'token ' + token,
        },
    )).read().decode()
except urllib.error.HTTPError as e:
    body = e.read().decode()  # Read the body of the error response

_json = json.loads(body)

ドキュメントHTTPErrorインスタンスを応答として使用する方法と、その他の属性について詳しく説明します。

8
Will Keeling