26

I've been trying to update a small Python library called libpynexmo to work with Python 3.

I've been stuck on this function:

def send_request_json(self, request):
    url = request
    req =  urllib.request.Request(url=url)
    req.add_header('Accept', 'application/json')
    try:
        return json.load(urllib.request.urlopen(req))
    except ValueError:
        return False

When it gets to this, json responds with:

TypeError: the JSON object must be str, not 'bytes'

I read in a few places that for json.load you should pass objects (In this case an HTTPResponse object) with a .read() attached, but it doesn't work on HTTPResponse objects.

I'm at a loss as to where to go with this next, but being that my entire 1500 line script is freshly converted to Python 3, I don't feel like going back to 2.7.

Chevron
  • 364
  • 2
  • 5
  • 10
  • 5
    See here for a solution: http://stackoverflow.com/questions/6862770/python-3-let-json-object-accept-bytes-or-let-urlopen-output-strings – dano Jun 05 '14 at 20:03
  • did you try passing it through 2to3? – zmo Jun 05 '14 at 20:03
  • @zmo - Did it manually so I could learn more. – Chevron Jun 05 '14 at 20:24
  • @dano - Found that link earlier, but was unable to make his workaround apply to my situation. I am unable to use .readall() on my HTTPResponse object. – Chevron Jun 05 '14 at 20:26
  • @Chevron Why can't you use `readall`? – dano Jun 05 '14 at 20:49
  • 2
    @Chevron, if you're trying to convert the json request object, then use this: json.loads(request.body.decode('utf-8')) – Anshuman Biswas May 28 '15 at 00:37
  • Possible duplicate of [Python 3, let json object accept bytes or let urlopen output strings](http://stackoverflow.com/questions/6862770/python-3-let-json-object-accept-bytes-or-let-urlopen-output-strings) – David Z May 16 '17 at 04:00

4 Answers4

47

Facing the same problem I solve it using decode()

...
rawreply = connection.getresponse().read()
reply = json.loads(rawreply.decode())
costas
  • 495
  • 1
  • 3
  • 3
  • 14
    Can you explain why this is necessary? – skaz Jun 26 '15 at 08:45
  • Check [this question](http://stackoverflow.com/questions/6862770/python-3-let-json-object-accept-bytes-or-let-urlopen-output-strings) for an explanation – J0ANMM Oct 04 '16 at 13:51
17

I recently wrote a small function to send Nexmo messages. Unless you need the full functionality of the libpynexmo code, this should do the job for you. And if you want to continue overhauling libpynexmo, just copy this code. The key is utf8 encoding.

If you want to send any other fields with your message, the full documentation for what you can include with a nexmo outbound message is here

Python 3.4 tested Nexmo outbound (JSON):

def nexmo_sendsms(api_key, api_secret, sender, receiver, body):
    """
    Sends a message using Nexmo.

    :param api_key: Nexmo provided api key
    :param api_secret: Nexmo provided secrety key
    :param sender: The number used to send the message
    :param receiver: The number the message is addressed to
    :param body: The message body
    :return: Returns the msgid received back from Nexmo after message has been sent.
    """


    msg = {
        'api_key': api_key,
        'api_secret': api_secret,
        'from': sender,
        'to': receiver,
        'text': body
    }
    nexmo_url = 'https://rest.nexmo.com/sms/json'
    data = urllib.parse.urlencode(msg)
    binary_data = data.encode('utf8')
    req = urllib.request.Request(nexmo_url, binary_data)
    response = urllib.request.urlopen(req)
    result = json.loads(response.readall().decode('utf-8'))
    return result['messages'][0]['message-id']
Aereaux
  • 845
  • 1
  • 8
  • 20
Ryan
  • 993
  • 1
  • 9
  • 17
13

I met the problem as well and now it pass

import json
import urllib.request as ur
import urllib.parse as par

html = ur.urlopen(url).read()
print(type(html))
data = json.loads(html.decode('utf-8'))
Du Peng
  • 351
  • 3
  • 3
3

Since you are getting a HTTPResponse, you can use Tornado.escape and its json_decode() to convert the JSON string into a dictionary:

from tornado import escape

body = escape.json_decode(body)

From the manual:

tornado.escape.json_decode(value)

Returns Python objects for the given JSON string.

jtlz2
  • 7,700
  • 9
  • 64
  • 114
fedorqui
  • 275,237
  • 103
  • 548
  • 598