Flask 中有没有办法获取 request.json 中的每个对象并将每个属性保存在模型上?
Is there a way in flask to get each object in the request.json and save each attribute on the model?
我有以下冗长的代码,它设置 request.json['example']
中的每个变量,然后保存模型 Inventory
。有没有办法用更少的代码来做到这一点?例如 ruby 我可以简单地将参数传递给 Inventory(inventory_params)
.
@api.route('/new/<id>', methods=['POST'])
@cross_origin()
def new_item(id):
date_opened = None
date_expired = None
received_date = None
order_date = None
if len(request.json['opened']) >= 1:
date_opened = datetime.strptime(request.json['opened'], '%m/%d/%Y')
print(date_opened)
if len(request.json['expiration_date']) >= 1:
date_expired = datetime.strptime(request.json['expiration_date'], '%m/%d/%Y')
if len(request.json['received_date']) >= 1:
received_date = datetime.strptime(request.json['received_date'], '%m/%d/%Y')
if len(request.json['order_date']) >= 1:
order_date = datetime.strptime(request.json['order_date'], '%m/%d/%Y')
amount=request.json['amount']
units=request.json['units']
location=request.json['location']
opened=request.json['opened']
lot=request.json['lot']
batch=request.json['batch']
notes=request.json['notes']
price=request.json['price']
country_code=request.json['country_code']
secondary_location=request.json['secondary_location']
vendor_code=request.json['vendor']
uid = request.json['uid']
item = Inventory(vendor_code=vendor_code, country_code=country_code, price=price, secondary_location=secondary_location, lot=lot, original_amount=amount, amount=amount, original_units=units, units=units, location=location, plant_material_id=id, uid=uid, batch=batch, notes=notes, date_opened=date_opened, date_expired=date_expired, received_date=received_date, order_date=order_date )
db.session.add(item)
db.session.commit()
return jsonify({'results': True})
您可以为此使用 argument unpacking。
基本原则是,如果你有一个接受关键字参数的函数或方法,比如
def foo(bar=42, qux=None):
pass
你可以不用像 foo(bar=1, qux=23)
那样调用它,而是从字典中获取这些参数,然后通过 解包 将它们传递给函数(这就是 **
语法确实如此):
kwargs = {'bar': 1, 'qux': 23}
foo(**kwargs)
因为在您的情况下,requests.json
中的 keys/values 与 Inventory
的构造函数所需的参数之间没有严格的 1:1 关系,您将需要对这些值进行一些预处理,因此
- 处理后从参数字典中删除原始值
- 使用正确的键将处理后的值添加到参数字典中
构建参数字典后,Inventory(**params)
会将 所有 这些参数传递给 Inventory.__init__()
,因此您需要确保匹配方法的签名.
这是一个独立的示例(没有 Flask 或 SQLAlchemy,但它确实没有区别):
from datetime import datetime
params = {
'opened': '05/07/2015',
'vendor_code': 'VC',
'country_code': 'US',
}
class Inventory(object):
def __init__(self, vendor_code=None, country_code=None, date_opened=None):
self.vendor_code = vendor_code
self.country_code = country_code
self.date_opened = date_opened
def __repr__(self):
return repr(self.__dict__)
def create_inventory(params):
# Deal with a raw value that needs to be processed
date_opened = None
if len(params['opened']) >= 1:
date_opened = datetime.strptime(params['opened'], '%m/%d/%Y')
# Remove the raw value from the params dict
params.pop('opened')
# Add the processed value to the `params` dict with the proper key
params['date_opened'] = date_opened
# All the other parameters get passed through unmodified
inv = Inventory(**params)
return inv
print create_inventory(params)
我有以下冗长的代码,它设置 request.json['example']
中的每个变量,然后保存模型 Inventory
。有没有办法用更少的代码来做到这一点?例如 ruby 我可以简单地将参数传递给 Inventory(inventory_params)
.
@api.route('/new/<id>', methods=['POST'])
@cross_origin()
def new_item(id):
date_opened = None
date_expired = None
received_date = None
order_date = None
if len(request.json['opened']) >= 1:
date_opened = datetime.strptime(request.json['opened'], '%m/%d/%Y')
print(date_opened)
if len(request.json['expiration_date']) >= 1:
date_expired = datetime.strptime(request.json['expiration_date'], '%m/%d/%Y')
if len(request.json['received_date']) >= 1:
received_date = datetime.strptime(request.json['received_date'], '%m/%d/%Y')
if len(request.json['order_date']) >= 1:
order_date = datetime.strptime(request.json['order_date'], '%m/%d/%Y')
amount=request.json['amount']
units=request.json['units']
location=request.json['location']
opened=request.json['opened']
lot=request.json['lot']
batch=request.json['batch']
notes=request.json['notes']
price=request.json['price']
country_code=request.json['country_code']
secondary_location=request.json['secondary_location']
vendor_code=request.json['vendor']
uid = request.json['uid']
item = Inventory(vendor_code=vendor_code, country_code=country_code, price=price, secondary_location=secondary_location, lot=lot, original_amount=amount, amount=amount, original_units=units, units=units, location=location, plant_material_id=id, uid=uid, batch=batch, notes=notes, date_opened=date_opened, date_expired=date_expired, received_date=received_date, order_date=order_date )
db.session.add(item)
db.session.commit()
return jsonify({'results': True})
您可以为此使用 argument unpacking。
基本原则是,如果你有一个接受关键字参数的函数或方法,比如
def foo(bar=42, qux=None):
pass
你可以不用像 foo(bar=1, qux=23)
那样调用它,而是从字典中获取这些参数,然后通过 解包 将它们传递给函数(这就是 **
语法确实如此):
kwargs = {'bar': 1, 'qux': 23}
foo(**kwargs)
因为在您的情况下,requests.json
中的 keys/values 与 Inventory
的构造函数所需的参数之间没有严格的 1:1 关系,您将需要对这些值进行一些预处理,因此
- 处理后从参数字典中删除原始值
- 使用正确的键将处理后的值添加到参数字典中
构建参数字典后,Inventory(**params)
会将 所有 这些参数传递给 Inventory.__init__()
,因此您需要确保匹配方法的签名.
这是一个独立的示例(没有 Flask 或 SQLAlchemy,但它确实没有区别):
from datetime import datetime
params = {
'opened': '05/07/2015',
'vendor_code': 'VC',
'country_code': 'US',
}
class Inventory(object):
def __init__(self, vendor_code=None, country_code=None, date_opened=None):
self.vendor_code = vendor_code
self.country_code = country_code
self.date_opened = date_opened
def __repr__(self):
return repr(self.__dict__)
def create_inventory(params):
# Deal with a raw value that needs to be processed
date_opened = None
if len(params['opened']) >= 1:
date_opened = datetime.strptime(params['opened'], '%m/%d/%Y')
# Remove the raw value from the params dict
params.pop('opened')
# Add the processed value to the `params` dict with the proper key
params['date_opened'] = date_opened
# All the other parameters get passed through unmodified
inv = Inventory(**params)
return inv
print create_inventory(params)