使用Python通过POST传递JSON对象

时间:2010-06-30 15:09:43

标签: python json google-app-engine post

我正在尝试通过POST发布JSON对象。我试着这样做:

import json, urllib, urllib2

filename = 'test.json'
race_id = 2530
f = open(filename, 'r')
fdata = json.loads(f.read())
f.close()

prefix = 'localhost:8000'

count = 0
for points in fdata['positions'].iteritems():
    print '--' + str(count) + '--------'
    url = 'http://'+prefix+'/api/points'
    parameters = {'point_data': json.dumps(points), 'race_id': race_id}
    data = urllib.urlencode(parameters)
    print data
    request = urllib2.Request(url, data)
    response = urllib2.urlopen(request)
    count += 1
    break;

print 'Finished adding points'

然后在另一端(我正在使用Google App Engine)收到数据:

point_data = json.load(self.request.get('point_data'))

但是我收到以下错误:

ERROR    2010-06-30 15:08:05,367
__init__.py:391] 'unicode' object has no attribute 'read' Traceback (most
recent call last):   File
"/home/ian/workspace/google_appengine/google/appengine/ext/webapp/__init__.py",
line 513, in __call__
    handler.post(*groups)   File "/home/ian/workspace/codebase/track_builder/geo-api.py",
line 276, in post
    point_data = json.load(self.request.get('point_data'))
File
"/home/ian/workspace/google_appengine/lib/django/django/utils/simplejson/__init__.py",
line 208, in load
    return cls(encoding=encoding, **kw).decode(fp.read()) AttributeError: 'unicode' object has
no attribute 'read' INFO    
2010-06-30 15:08:05,376
dev_appserver.py:3266] "POST
/api/points HTTP/1.1" 500 -

有关如何解决此问题的任何想法?

编辑:根据要求,这里是一个例子:

(u'1276859700',
{
    u'24': {
        u'tempc': u'-22.7',
        u'gpsq': u'1',
        u'altm': u'65527',
        u'hd': u'112',
        u'hdop': u'0.93',
        u'bton': u'0',
        u'maxv': u'20.15',
        u'idit': u'1',
        u'satc': u'10',
        u'minv': u'20.15',
        u'lat': u'35.271993',
        u'btusr': u'0',
        u'lng': u'-121.845353',
        u'knots': u'7'
    },
    u'11': {
        u'tempc': u'13.0',
        u'gpsq': u'1',
        u'altm': u'65535',
        u'hd': u'130',
        u'hdop': u'0.84',
        u'bton': u'0',
        u'maxv': u'15.96',
        u'idit': u'1',
        u'satc': u'12',
        u'minv': u'15.88',
        u'lat': u'34.877815',
        u'btusr': u'0',
        u'lng': u'-121.386116',
        u'knots': u'8'
    }
}

编辑2: 感谢Daniel Roseman和Nick Johnson,他们都抓住了我的错误。我改变了

point_data = json.loads(self.request.get('point_data'))

这解决了错误,但现在我得到了:

ERROR    2010-06-30 16:07:29,807 __init__.py:391] 'list' object has no attribute 'iteritems'
Traceback (most recent call last):
  File "/home/ian/workspace/google_appengine/google/appengine/ext/webapp/__init__.py", line 513, in __call__
    handler.post(*groups)
  File "/home/ian/workspace/codebase/track_builder/geo-api.py", line 255, in post
    for time, units in point_data.iteritems():
AttributeError: 'list' object has no attribute 'iteritems'
INFO     2010-06-30 16:07:29,816 dev_appserver.py:3266] "POST /api/points HTTP/1.1" 500 -

涉及以下代码:

class TrackPoint(webapp.RequestHandler):
    def post(self):
        point_data = json.loads(self.request.get('point_data'))
        race_id = self.request.get('race_id')
        added = []
        failed = []
        for time, units in point_data.iteritems():
            for unit, data in units.iteritems():
                ...

关于这个的任何想法?

3 个答案:

答案 0 :(得分:31)

看起来self.request.get()正在返回一个unicode对象,而不是一个类似文件的对象。您可以尝试使用json.loads()代替json.load()

答案 1 :(得分:4)

json.load()需要一个文件对象,但self.request.get以字符串形式返回参数的值。

解决方案很简单:使用json.loads。

此外,免费提示:我从名称中假设您正在捆绑自己的json库副本。 App Engine实际上包含了一个你可以使用的simplejson副本 - 只需:

from django.utils import simplejson

答案 2 :(得分:1)

首先在js中我认识到json对象必须采用STRING格式(javascript文件)

// using jquery, json2
var js = {"name":"nguyen","age":"1"};
$.post("/", {'data': JSON.stringify(js)}, function(ret){
        alert(ret);
    });

然后在gae

from django.utils import simplejson as json

class PesonReq(webapp.RequestHandler):
    def post(self):
        t = json.loads(self.request.POST['data'])
        self.response.out.write(t['name'])