簡體   English   中英

Django:使用True True來增加Slug的大小

[英]Django: Increasing the size of Slug with Unique True

我需要將子彈的大小增加到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)

請指教。

根據文檔SlugField的最大長度為255個字符。 但是數據庫后端強制執行此操作,因此,根據您使用的后端,您也許可以增加它。

SQLite和PostgreSQL都允許使用UNIQUE約束將max_length設置為500。 請注意,PostgreSQL實際上會強制執行長度限制,而SQLite並不在乎,它允許您插入大於聲明大小的字符串。

但是MySQL / Mariadb驅動程序將在django/db/backends/mysql/validation.py強制使用255個字符的限制。 在標准后端中,這是唯一具有此限制的后端。 255來自InnoDB列的最大索引鍵為767個字節的限制。 如果必須將UTF8數據存儲在該列中,則每個字符最多可占用3個字節,這將導致767/3 = 255個字符。

一種可能的解決方案是使用一個額外的CharField其中包含該段的哈希值。 該字段將很短(例如,用於MD5摘要的32個字節) 並且它將是唯一的,這將阻止插入重復的段。 塊的填充及其哈希可以在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)

現在,如果嘗試使用相同的段名來保存記錄,則會在slug_hash上引發IntegrityError

是的,可以使用信號。 在保存模型實例之前,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 = models.SlugField(max_length = 255, unique=True)

如果更改數據庫技術,請不要超過255,以免造成麻煩。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM