我需要将我的slug的大小增加到500,默认情况下是255个字符,但同样我需要保持它的独特性。有没有办法实现它?
models.py
class Product(models.Model):
title = models.CharField(max_length=500)
description = models.TextField(blank=True, null=True)
price = models.DecimalField(max_digits=20, decimal_places=2)
sku = models.CharField(null=True, max_length=100)
url = models.URLField(blank=True)
slug = models.SlugField(unique=True)
请告知。
答案 0 :(得分:2)
根据documentation,SlugField
的最大长度为255个字符。但是正是数据库后端强制执行它,这取决于您使用的是哪个后端,可能能够增加它。
SQLite和PostgreSQL都允许max_length
使用UNIQUE
约束设置为500。请注意,PostgreSQL实际上会强制执行长度限制,而SQLite并不真正关心,并允许您插入大于声明大小的字符串。
但MySQL / Mariadb驱动程序将在django/db/backends/mysql/validation.py
中强制执行255个字符的限制。在标准后端中,这是唯一具有此限制的后端。 255来自InnoDB列的最大索引键为767字节的限制。如果您在列中存储UTF8数据,则必须使用每个字符最多3个字节,这将导致767/3 = 255个字符。
一种可能的解决方案是让一个额外的CharField
包含一个slug的哈希值。该字段将很短(例如,MD5摘要的32个字节)和它将是唯一的,这将阻止插入的重复slugs。 slug的种群及其哈希可以在Product.save()
中完成。
模型将是:
import hashlib
from django.utils.text import slugify
class Product(models.Model):
title = models.CharField(max_length=500)
description = models.TextField(blank=True, null=True)
price = models.DecimalField(max_digits=20, decimal_places=2)
sku = models.CharField(null=True, max_length=100)
url = models.URLField(blank=True)
slug = models.SlugField(max_length=500)
slug_hash = models.CharField(max_length=32, unique=True)
def save(self, *args, **kwargs):
self.slug = slugify(self.title)
self.slug_hash = hashlib.md5(self.slug).hexdigest()
super(Product, self).save(*args, **kwargs)
现在,如果尝试使用相同的段塞保存记录,则会IntegrityError
上的slug_hash
被提升。
答案 1 :(得分:0)
是的,可以使用信号。在保存模型实例之前,将自动创建/更新slug字段。要使其唯一,如果已存在具有相同slug的模型实例,则可以将标题与id连接。请参阅下面的代码(将其添加到您的模型中):
def create_slug(instance, new_slug=None):
"""
A function to create a slug
if an object with slug already exists, it adds id value to slug
else just slugify the title
"""
slug = slugify(instance.title)
if new_slug is not None:
slug = new_slug
qs = Product.objects.filter(slug=slug).order_by('-id')
exists = qs.exists()
if exists:
new_slug = "%s-%s" %(slug, qs.first().id)
return create_slug(instance, new_slug=new_slug) #recursive call
return slug
def pre_save_product_receiver(sender, instance, *args, **kwargs):
"""
A signal that calls the function create_slug if an object has not slug
"""
if not instance.slug:
instance.slug = create_slug(instance)
pre_save.connect(pre_save_product_receiver, sender=Product)
要增加slug max字符,请用以下字符替换字段:
slug = models.SlugField(max_length = 255, unique=True)
请不要超过255,以免在更改数据库技术时出现问题。