2014-05-02 74 views
2

我需要從燒瓶中創建一個json對象查詢結果。然後我需要將json對象傳遞給路由來創建一個API。燒瓶:將Python字典轉換爲客戶端api的json對象

在尋找一些流暢的方法來創建我的實例中的字典時,我偶然發現了這個post中的一個方法來使用實例的內部字典並向模型類添加一個jsond方法。下面是用自定義的方法模型,「jsond」:

from app import db 
class Rest(db.Model): 
    id = db.Column(db.Integer, primary_key = True) 
    name = db.Column(db.String(100), unique = True) 
    street = db.Column(db.Text) 
    zipcd = db.Column(db.Integer) 
    comments = db.relationship('Comment', backref='rest', lazy='dynamic') 
    lat = db.Column(db.Float(6)) 
    lng = db.Column(db.Float(6)) 

    def __init__(self,name,street,zipcd): 
     self.name = name 
     self.street = street 
     self.zipcd = zipcd 

    def __repr__(self): 
     return '{}'.format(self.name) 

    def name_slug(self): 
     return self.name 

    def jsond(self): 
     instDict = self.__dict__.copy() 
     if instDict.has_key('_sa_instance_state'): 
      del instDict['_sa_instance_state'] 
     return instDict 

這是我的看法功能:

from app import app, db 
from flask import render_template, flash, redirect, session, url_for, request, g,  jsonify, make_response 
from flask.json import dumps 
from flask.ext import restful 
from flask.ext.httpauth import HTTPBasicAuth 
from models import Comment, Rest, Badge 
from helper import make_badges, make_inspections, loc_query 
import operator 
auth = HTTPBasicAuth() 

@app.route('/api',methods=['GET']) 
def makeApi(): 

    ###Query Parameters### 
    lim = request.args.get('limit', 10) 
    off = request.args.get('offset', 0) 
    loc = request.args.get('location', "39.94106,-75.173192") 
    lat, lng = loc.split(",") 
    radius = request.args.get('radius',2) 

    query = loc_query(lat,lng,radius,off,lim) 

    results = Rest.query.from_statement(query).all() 


    rest_json = [] 
    for rest in results: 
     rest_json.append(rest.jsond()) 

    return make_response(jsonify({'count':len(rest_json),'rests':rest_json})) 

所以當在Python API命令行我可以成功運行查詢和創建的字典從具有所有字段的查詢中的特定實例(使用自定義jsond方法)。然而,當我用我的看法轉到makeApi路線,我得到一個JSON對象僅與「ID」字段存在:

rests: [ 
{ 
id: 28450 
}, 
{ 
id: 28795 
}, 
{ 
id: 30439 
}, 
{ 
id: 29325 
}, 
{ 
id: 29765 
}, 
{ 
id: 29928 
}, 
{ 
id: 30383 
}, 
{ 
id: 29064 
}, 
{ 
id: 29862 
}, 
{ 
id: 28610 
} 
] 
} 

我在圈子裏已經持續了幾個小時,而且不知道爲什麼視圖的行爲會與python API不同。也許它的東西我做錯了jsonify,但我不這麼認爲。

+0

「轉儲({...})」,你已經導入了什麼? – swstephe

+0

@swstephe轉儲給我同樣的錯誤 - ony'id'字段顯示 – user2957824

回答

2

通常是一個壞主意,你對jsonifying模型的方法:

  • self.__dict__可能含有大量的無證鍵
  • 列類型:你不能直接jsonify relationship和列類型。特別是datetime列。
  • 安全:有時候你可能想隱藏某些字段(不受信任的用戶消耗你比如API)的

一個好的方法是創建一個方法返回一個JSON序列化的詞典:

class Foo(db.Model): 
    field1 = db.Column(...) 
    field2 = db.Column(...) 

    def as_dict(self): 
     obj_d = { 
      'field1': self.field1, 
      'field2': self.field2, 
      ... 
     } 
     return obj_d 

然後在您的視圖:

foos = Foo.query.all() 
results = [ foo.as_dict() for foo in foos ] 

return jsonify({count: len(results), results: results) 

根據您的應用程序,你可以通過轉換領域做出更明智的as_dict(特別是DAT etime fileds)以javascript友好的格式或添加方便的領域,如下面的關係。

+0

作品像魅力謝謝! – user2957824

+0

@paolocasiello有沒有辦法傳遞一個字典對象在模板中直接使用?如果是的話,這是否有益? – user2957824

+0

你的意思是Jinja模板?您可以將整個模型傳遞給'render_template(...,mod = model)'模板,並直接使用模板中的函數/屬性 –