Django REST 框架验证错误:'Enter a valid URL.'



在我的Django REST框架项目中,我有一个模型类,用于保存Django应用程序将在后台任务中抓取的服务:

class Service(models.Model):
name = models.CharField(max_length=50)
description = models.CharField(max_length=250)
root_url =URLField(unique=True)

早些时候,我在本地机器上运行了这些服务和 Django 应用程序。现在我容器化了服务和 Django 应用程序,并让它们在 Docker 上运行。

所以现在我在添加码头服务时遇到了问题,因为它的根 URL:http://sensor-service:8080/.引发的错误是:'Enter a valid URL.'

当向我的 Django REST API 进行 POST 时,以及在通过 Django REST 框架提供的 GUI 添加服务时,都会出现问题。

因此,基于 https://stackoverflow.com/a/49262127/5433896 和 http://www.django-rest-framework.org/api-guide/serializers/#field-level-validation,我尝试了以下内容。注意:最重要的一行是第 5 行:允许主机名后面不跟域和/或 TLD。

models.py:

class DockerServiceDNSUrlsCompatibleURLValidator(URLValidator):
def __init__(self, schemes=None, **kwargs):
self.host_re = '(' + self.hostname_re + self.domain_re + self.tld_re 
+ '|localhost|' 
+ self.hostname_re + ')'  # <=== added: "hostname not followed by a domain and/or TLD is acceptable"
self.regex = _lazy_re_compile(r'^(?:[a-z0-9.-+]*)://'  # scheme is validated separately
r'(?:S+(?::S*)?@)?'  # user:pass authentication
r'(?:' + self.ipv4_re + '|' + self.ipv6_re + '|' + self.host_re + ')'
r'(?::d{2,5})?'  # port
r'(?:[/?#][^s]*)?'  # resource path
r'Z',
re.IGNORECASE)
super().__init__(schemes, **kwargs)

class DjangoServiceDNSUrlsCompatibleFormURLField(forms.fields.URLField):
default_validators = [DockerServiceDNSUrlsCompatibleURLValidator()]

class DjangoServiceDNSUrlsCompatibleURLField(models.URLField):
default_validators = [DockerServiceDNSUrlsCompatibleURLValidator()]
def formfield(self, **kwargs):
return super(DjangoServiceDNSUrlsCompatibleURLField, self).formfield(**{
'form_class': DjangoServiceDNSUrlsCompatibleFormURLField,
})

class Service(models.Model):
name = models.CharField(max_length=50)
description = models.CharField(max_length=250)
root_url = DjangoServiceDNSUrlsCompatibleURLField(unique=True)

serializers.py:

class ServiceSerializer(serializers.HyperlinkedModelSerializer):
validators = [DockerServiceDNSUrlsCompatibleURLValidator()]
def validate_root_url(self, value):
DockerServiceDNSUrlsCompatibleURLValidator(value)  # will throw validation exception if value isn't OK.
return value

但是,仍然会引发错误'Enter a valid URL.'。有人看到我的方法有问题吗?或者任何人 - 最坏的情况,但仍然可以接受 - 知道如何在此字段上完全关闭 URLValidation?

如果你有自己的正则表达式,为什么不直接使用模型。CharField 而不是 URLField?喜欢:

models.py

phone_regex = RegexValidator(
regex=r'^+?(?:(?[0-9])?x20?){4,14}[0-9]$',
message="Error....")
class FooModel(models.Model):
phone = models.CharField("Phone Number", validators=[
phone_regex], max_length=26, blank=True, null=True)

顺便说一句,要自定义URLValidator接受没有"http/https"的URL,我在下面使用

models.py

class OptionalSchemeURLValidator(URLValidator):
def __call__(self, value):
if '://' not in value:
# Validate as if it were http://
value = 'http://' + value
super(OptionalSchemeURLValidator, self).__call__(value)
class FooModel(models.Model):
website = models.CharField(max_length=200, validators=[OptionalSchemeURLValidator()])

这不会更改上传的值,只需使验证通过即可。 我试过我的 DRF,它有效。

参考阿拉斯代尔的回答,谢谢阿拉斯代尔

相关内容

最新更新