Как создать уникального слизняка в Django
Я пытаюсь создать уникальный слизень в Django, чтобы я мог получить доступ к сообщению через url-адрес, подобный этому: http://www.example.com/buy-a-new-bike_Boston-MA-02111_2
соответствующие модели:
class ZipCode(models.Model):
zipcode = models.CharField(max_length=5)
city = models.CharField(max_length=64)
statecode = models.CharField(max_length=32)
class Need(models.Model):
title = models.CharField(max_length=50)
us_zip = models.CharField(max_length=5)
slug = ?????
def get_city():
zip = ZipCode.objects.get(zipcode=self.us_zip)
city = "%s, %s %s" % (zip.city, zip.statecode, zip.zipcode)
return city
пример записи ZipCode:
- zipcode = "02111"
- город = "Бостона"
- statecode = "MA"
запись потребности образца:
- title = " купить новый велосипед"
- us_zip = "02111"
- slug = "buy-a-new-bike_Boston-MA-02111_2" (желательно)
любые советы о том, как создать этот уникальный слаг? Его состав:
- нужно.название + " _ " + нужно.get_city () + " _ " + необязательное инкрементное целое число, чтобы сделать его уникальным. Все пробелы следует заменить на" -".
Примечание: мой желаемый слизень выше предполагает, что слизень "buy-a-new-bike_Boston-MA-02111" уже существует, который это то, что к нему прилагается" _2", чтобы сделать его уникальным.
Я пробовал Django-extensions, но кажется, что для создания уникального слизняка может потребоваться только поле или кортеж полей. Мне нужно передать функцию get_city (), а также соединитель "_" между заголовком и городом. Кто-нибудь решил это и готов поделиться?
спасибо!
обновление
Я уже использую Django-extensions для своего UUIDField, поэтому было бы неплохо если он также может быть использован для его AutoSlugField!
7 ответов
Я использую этот фрагмент для генерации уникального слизняка и моего типичного метода сохранения выглядит ниже
slug будет Django SlugField с blank=True, но принудительно slug в методе сохранения.
типичный метод сохранения для модели Need может выглядеть ниже
def save(self, **kwargs):
slug_str = "%s %s" % (self.title, self.us_zip)
unique_slugify(self, slug_str)
super(Need, self).save(**kwargs)
и это будет генерировать слизень, как buy-a-new-bike_Boston-MA-02111 , buy-a-new-bike_Boston-MA-02111-1 и так далее. Выходные данные могут немного отличаться, но вы всегда можете пройти фрагмент и подгоняйте к вашим потребностям.
мой маленький код:
def save(self, *args, **kwargs):
strtime = "".join(str(time()).split("."))
string = "%s-%s" % (strtime[7:], self.title)
self.slug = slugify(string)
super(Need, self).save()
Если вы думаете об использовании приложения, чтобы сделать это для вас, вот один.
https://github.com/un33k/django-uuslug
UUSlug = (``U``nique + ``U``code Slug)
Unicode Test Example
=====================
from uuslug import uuslug as slugify
s = "This is a test ---"
r = slugify(s)
self.assertEquals(r, "this-is-a-test")
s = 'C\'est déjà l\'été.'
r = slugify(s)
self.assertEquals(r, "c-est-deja-l-ete")
s = 'Nín hǎo. Wǒ shì zhōng guó rén'
r = slugify(s)
self.assertEquals(r, "nin-hao-wo-shi-zhong-guo-ren")
s = '影師嗎'
r = slugify(s)
self.assertEquals(r, "ying-shi-ma")
Uniqueness Test Example
=======================
Override your objects save method with something like this (models.py)
from django.db import models
from uuslug import uuslug as slugify
class CoolSlug(models.Model):
name = models.CharField(max_length=100)
slug = models.CharField(max_length=200)
def __unicode__(self):
return self.name
def save(self, *args, **kwargs):
self.slug = slugify(self.name, instance=self)
super(CoolSlug, self).save(*args, **kwargs)
Test:
=====
name = "john"
c = CoolSlug.objects.create(name=name)
c.save()
self.assertEquals(c.slug, name) # slug = "john"
c1 = CoolSlug.objects.create(name=name)
c1.save()
self.assertEquals(c1.slug, name+"-1") # slug = "john-1"
это простая реализация, которая генерирует Слизень из заголовка, он не зависит от других фрагментов:
from django.template.defaultfilters import slugify
class Article(models.Model):
...
def save(self, **kwargs):
if not self.slug:
slug = slugify(self.title)
while True:
try:
article = Article.objects.get(slug=slug)
if article == self:
self.slug = slug
break
else:
slug = slug + '-'
except:
self.slug = slug
break
super(Article, self).save()
Django предоставляет поле модели SlugField, чтобы сделать это проще для вас. Вот пример этого в приложении" блог"
class Post(models.Model):
title = models.CharField(max_length=100)
content = models.TextField(blank=True)
slug = models.SlugField(unique=True)
@models.permalink
def get_absolute_url(self):
return 'blog:post', (self.slug,)
обратите внимание, что мы установили unique=True для нашего поля slug - в этом проекте мы будем искать сообщения по их slug, поэтому нам нужно убедиться, что они уникальны. Вот что наше приложение views.py может выглядеть так:
from .models import Post
def post(request, slug):
post = get_object_or_404(P`enter code here`ost, slug=slug)
return render(request, 'blog/post.html', {
'post': post,
})
привет, вы можете попробовать эту функцию
class Training(models.Model):
title = models.CharField(max_length=250)
text = models.TextField()
created_date = models.DateTimeField(
auto_now_add=True, editable=False, )
slug = models.SlugField(unique=True, editable=False, max_length=250)
def __unicode__(self):
return self.title
def save(self, *args, **kwargs):
self.slug =get_unique_slug(self.id,self.title,Training.objects)
return super(Training, self).save(*args, **kwargs)
def get_unique_slug(id,title,obj):
slug = slugify(title.replace('ı', 'i'))
unique_slug = slug
counter = 1
while obj.filter(slug=unique_slug).exists():
if(obj.filter(slug=unique_slug).values('id')[0]['id']==id):
break
unique_slug = '{}-{}'.format(slug, counter)
counter += 1
return unique_slug
class Need(models.Model):
title = models.CharField(max_length=50)
us_zip = models.CharField(max_length=5)
slug = models.SlugField(unique=True)
def save(self, **kwargs):
slug_str = "%s %s" % (self.title, self.us_zip)
super(Need, self).save()