【问题标题】:Python/API: Handling an error that the server doesn't returnPython/API:处理服务器不返回的错误
【发布时间】:2018-12-12 04:19:03
【问题描述】:

我正在努力处理 Outbrain API 服务器未返回的错误。我所做的是向带有国家名称的服务器发送一个 GET 请求,它返回一个包含所需 ID 值的 JSON 响应,但是当我输入带有错字的国家名称时,JSON 响应是空的,而不是带有错误的 JSON消息(如果我没记错的话)。所以我试图自己处理错误,但它不起作用。这是我的代码(其中 csv[row][2] = United Kinjdom):

r = requests.get('https://api.outbrain.com/amplify/v0.1/locations/search?term=' + csv[row][2] + '&limit=7', headers=headers)
try:
    print(r.json())
    data = r.json()
    error()
except:
    print(r.text)
    data = r.text
    error()

for results in data:
    if results['geoType']:
        if results['geoType'] == 'Country' and results['name'] == csv[row][2]:
            print(results['id'])
            countryId = results['id']
    else:
        logText = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime()) + ": Country name error"
        c.execute("INSERT INTO log VALUES (?)", [logText])
        conn.commit()
        os._exit(1)

第一次打印返回“[]”,代码似乎没有到达“else”语句,所以我可以将错误添加到日志数据库中。

当国家拼写正确时服务器响应:

[{'id': 'c20768bb56a25c22299c38801e935c3a', 'geoType': 'Country', 'name': 'United Kingdom', 'canonicalName': 'United Kingdom', 'code': 'GB'},
{'id': '7477c333ed4e1895b040efe45c30c816', 'geoType': 'Region', 'name': 'Darlington', 'canonicalName': 'Darlington, United Kingdom', 'code': 'D1', 'parent': {'id': 'c20768bb56a25c22299c38801e935c3a', 'geoType': 'Country', 'name': 'United Kingdom', 'canonicalName': 'United Kingdom', 'code': 'GB'}},
{'id': 'd9a9732283ec131e7fa422843449baa4', 'geoType': 'Region', 'name': 'Bath and North East Somerset', 'canonicalName': 'Bath and North East Somerset, United Kingdom', 'code': 'A4', 'parent': {'id': 'c20768bb56a25c22299c38801e935c3a', 'geoType': 'Country', 'name': 'United Kingdom', 'canonicalName': 'United Kingdom', 'code': 'GB'}}, 
{'id': '92b5f5ef7a5a25e60263e365982be9ad', 'geoType': 'Region', 'name': 'Northumberland', 'canonicalName': 'Northumberland, United Kingdom', 'code': 'J6', 'parent': {'id': 'c20768bb56a25c22299c38801e935c3a', 'geoType': 'Country', 'name': 'United Kingdom', 'canonicalName': 'United Kingdom', 'code': 'GB'}},
{'id': '3a83dddcd55e92aaab4b142c7858892d', 'geoType': 'Region', 'name': 'Vale of Glamorgan', 'canonicalName': 'Vale of Glamorgan, United Kingdom', 'code': 'Z3', 'parent': {'id': 'c20768bb56a25c22299c38801e935c3a', 'geoType': 'Country', 'name': 'United Kingdom', 'canonicalName': 'United Kingdom', 'code': 'GB'}},
{'id': '915c6d5814a9826e34e1a5c0a423797a', 'geoType': 'Region', 'name': 'Walsall', 'canonicalName': 'Walsall, United Kingdom', 'code': 'O8', 'parent':                     {'id': 'c20768bb56a25c22299c38801e935c3a', 'geoType': 'Country', 'name': 'United Kingdom', 'canonicalName': 'United Kingdom', 'code': 'GB'}},
{'id': 'e96bcc2de342e74e491d3b6ab95cfedc', 'geoType': 'Region', 'name': 'Tameside', 'canonicalName': 'Tameside, United Kingdom', 'code': 'O1', 'parent': {'id': 'c20768bb56a25c22299c38801e935c3a', 'geoType': 'Country', 'name': 'United Kingdom', 'canonicalName': 'United Kingdom', 'code': 'GB'}}]

【问题讨论】:

  • 在发送请求之前验证请求不是更聪明吗...?
  • 脚本不仅要考虑每个国家,还要考虑每个城市和 DMA,这样编码起来很麻烦
  • 我认为你想移动你的 else 语句或在你的迭代中添加一个辅助 else ,所以如果名称不匹配,它会返回一个错误,但你应该按照上面的说明来重构它。跨度>

标签: python api


【解决方案1】:

根据您的问题,您的代码可能只执行tryexcept 部分。由于您的第一个print 给出了输出[],那么服务器可能什么也不返回,甚至没有警告消息。由于您的数据是空列表,因此不会执行 for 循环。我担心的是,除了这里之外,您要尝试的 Exception 是什么。如果您需要处理服务器返回的结果以防您输入错误,这意味着您收到一个空列表,那么我建议:

data = r.json()    
if not data: raise Exception

然后将此代码块放入 except 处理部分,而不是将它们放入 else 部分(如果这是您希望在 Exception 发生时发生的事情)。

logText = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime()) + ": Country name error"
c.execute("INSERT INTO log VALUES (?)", [logText])
conn.commit()
os._exit(1)

此外,您正在使用if results['geoType'],只有当'geoType'键的值不是None时才等于if True,那么else部分应该只在该值是@时执行987654334@。这有点危险,因为如果密钥不在结果中,那么它将立即引发Exception。我可能会建议你在这里应该更具体一点,因为这可能会让你头疼。

【讨论】:

  • 我的错,我在验证错误的东西。谢谢你的解释,这很有帮助。我唯一不明白的是为什么服务器返回一个空列表而不是警告消息。我想知道这是否不是我在请求中犯的错误。
  • 我认为这是因为 API 没有进行检查以正确处理请求。在这种情况下,可能服务器没有返回任何内容,或者服务器端不接受请求,然后他们决定忽略它。
  • 这是我的第一个猜测。感谢您的验证。
【解决方案2】:

你应该检查你的结果列表(data)是否不为空,在这种情况下你可以迭代它,否则你不能,这就是为什么你从不输入你的 else 子句。

r = requests.get('https://api.outbrain.com/amplify/v0.1/locations/search?term=' + csv[row][2] + '&limit=7', headers=headers)
try:
    print(r.json())
    data = r.json()
    error()
except:
    print(r.text)
    data = r.text
    error()

if data:    
    for result in data:
        if result.get('geoType') == 'Country' and result['name'] == csv[row][2]:
            print(result['id'])
            countryId = result['id']
else:
    logText = time.strftime('%Y-%m-%d %H:%M:%S', time.localtime()) + ": Country name error"
    c.execute("INSERT INTO log VALUES (?)", [logText])
    conn.commit()
    os._exit(1)

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-10-13
    • 1970-01-01
    • 1970-01-01
    • 2023-03-21
    相关资源
    最近更新 更多