当我创建一个具有外部关系的新资源(指定为{"pk": 20}
)时,我会创建一个新的不需要的FK项。
我的Order
模型类与Language
模型有关系,因此在创建Order实例时,我可能需要指定订单的语言。语言列表应该是不变的,用户不能具有修改existant或创建新语言的能力。
Order
资源:
class OrderResource(ModelResource):
user = fields.ForeignKey(UserResource, 'user', null=True, full=True)
src_lang = fields.ForeignKey(LanguageResource, 'src_lang', null=True, full=True)
dst_lang = fields.ForeignKey(LanguageResource, 'dst_lang', null=True, full=True)
def obj_create(self, bundle, request=None, **kwargs):
return super(OrderResource, self).obj_create(bundle, request, user=request.user)
class Meta:
resource_name = 'orders'
queryset = Order.objects.all()
serializer = Serializer(['json'])
authentication = MultiAuthentication(SessionAuthentication(), ApiKeyAuthentication())
authorization = ResourceAuthorization()
这是一个Language
资源:
class Language(models.Model):
name = models.CharField(max_length=100)
code = models.CharField(max_length=100)
class LanguageResource(ModelResource):
class Meta:
resource_name = 'languages'
queryset = Language.objects.all()
allowed_methods = ['get']
authorization = ReadOnlyAuthorization()
serializer = Serializer(['json'])
我正在尝试使用jQuery创建一个新的Order
:
var data = JSON.stringify({
"comment": "Something random",
"src_lang": {"pk": "20"},
"dst_lang": "/api/v2/languages/72/"
});
$.ajax({
type: 'POST',
url: '/api/v2/orders/',
data: data,
dataType: "json",
contentType: "application/json"
});
不是将pk:20
设置为src_lang_id
字段,而是为Language
创建一个包含空字段的新src_lang
,并为dst_lang
设置正确的值。但是使用Language
模型定义限制空字段。如何保存它?
这也很奇怪,因为我直接指定了语言模型的只读访问权限,并且只有get
方法来访问支持的语言列表。
如果我将OrderResource
类的语言字段声明为:src_lang = fields.ForeignKey(LanguageResource, 'src_lang', null=True, full=True, readonly=True)
,则它不会创建任何内容,但也不会为外键设置任何值。
所以,我只需要指定一种语言,我不需要创建它。
ResourceAuthorization
:
class ResourceAuthorization(Authorization):
def is_authorized(self, request, object=None):
user = getattr(request, 'user', None)
if not user:
return False
return user.is_authenticated()
def apply_limits(self, request, object_list):
if request and hasattr(request, 'user'):
if request.user.is_superuser:
return object_list
return object_list.filter(user=request.user)
return object_list.none()
我发现没有更聪明的制作字段只读并覆盖obj_create
方法:
class OrderResource(ModelResource):
user = fields.ForeignKey(UserResource, 'user', null=True, full=True)
src_lang = fields.ForeignKey(LanguageResource, 'src_lang', null=True, full=True, blank=True, readonly=True)
dst_lang = fields.ForeignKey(LanguageResource, 'dst_lang', null=True, full=True, blank=True, readonly=True)
def obj_create(self, bundle, request=None, **kwargs):
src_lang_id, dst_lang_id = bundle.data.get('src_lang', None), bundle.data.get('dst_lang', None)
if not all([src_lang_id, dst_lang_id]):
raise BadRequest('You should specify both source and destination language codes')
src_lang, dst_lang = Language.objects.guess(src_lang_id), Language.objects.guess(dst_lang_id)
if not all([src_lang, dst_lang]):
raise BadRequest('You should specify both source and destination language codes')
return super(OrderResource, self).obj_create(
bundle, request, user=request.user, src_lang=src_lang, dst_lang=dst_lang
)
class Meta:
resource_name = 'orders'
queryset = Order.objects.all()
serializer = Serializer(['json'])
authentication = MultiAuthentication(SessionAuthentication(), ApiKeyAuthentication())
authorization = ResourceAuthorization()
答案 0 :(得分:2)
正如this对您的问题的回答所述,src_lang
应该与资源相对应,而不是与其他某些值相对应。我怀疑当发生POST并且找不到资源pk=20
时,它会创建一个新的Language
对象并在没有Django模型验证的情况下调用save
,允许存在空白字段创建的Language
。
强制只读类型资源的一种方法是创建一个不允许obj_create
的资源。
class ReadOnlyLanguageResource(ModelResource):
# All the meta stuff here.
def obj_create(self):
# This should probably raise some kind of http error exception relating
# to permission denied rather than Exception.
raise Exception("Permission denied, cannot create new language resource")
然后从Order
资源引用此资源,仅覆盖src_lang
字段以指向您的只读资源。
class OrderResource(ModelResource):
user = fields.ForeignKey(UserResource, 'user', null=True, full=True)
src_lang = fields.ForeignKey(ReadOnlyLanguageResource, 'src_lang')
dst_lang = fields.ForeignKey(ReadOnlyLanguageResource, 'dst_lang')
任何引用现有资源的请求都将按照正常情况完成(但您需要正确引用资源,而不是使用pk=20
)。
Language
对象,因此任何请求引用未知语言的请求都将失败。
答案 1 :(得分:1)
您应该使用与src_lang
对应的/api/v2/languages/72/
格式指定pk=20
。
其次究竟是什么ResourceAuthorization
? documentation列出ReadOnlyAuthorization
可能对您有用。
此外,授权也适用资源,而不是基础模型。为fk
创建新对象时,它不使用REST Api,而是使用django.db.models
及其权限。因此授权可能不适用于外键约束。