File uploads в django 3
ДА! Это наконец-то случилось! Иван может петь победоносную песню, он так долго боролся за решение этого вопроса!
By default, if an uploaded file is smaller than 2.5 megabytes, Django will hold the entire contents of the upload in memory. This means that saving the file involves only a read from memory and a write to disk and thus is very fast. However, if an uploaded file is too large, Django will write the uploaded file to a temporary file stored in your system’s temporary directory.
Полная дока по новому аплоаду — Django | File Uploads | Django Documentation.
Две модели в newforms 10
На днях нужно было за короткое время решить одну несложную и довольно типичную задачу на django — построение формы профайла пользователя. Я не стал искать в закладках ссылки на старые известные how-to от мэтров-джангистов :-), а попробовал посмотреть что же нового и интересного у нас имеется в последней версии django (trunk).
Итак, модель была приготовлена заранее, вопрос больше касался построения формы (через newforms) пользовательского профайла. Итак, модель.
class UserProfile(models.Model):
user = models.ForeignKey(User, verbose_name=_('user'), unique=True, related_name="profile")
country = models.ForeignKey(Country, verbose_name=_('country'), blank=True, null=True)
city = models.CharField(verbose_name=_('city'), max_length=30, blank=True, null=True)
post_zip = models.CharField(verbose_name=_('post zip'), max_length=10, blank=True)
address = models.CharField(verbose_name=_('address'), max_length=250, blank=True)
Обратите внимание, что профайл ссылается на модель User с unique=True. Это в принципе самый простой и работающий вариант расширения модели пользователя в django. Можно смело использовать для объекта User метод profile.get(). Например:
from django.contrib.auth.models import User
u = User.objects.get(username='test')
profile = u.profile.get()
Теперь для моделей User и UserProfile необходимо построить единую форму. Для каждой отдельной модели сделать это не проблема. А вот сделать «два в одном» не так просто. Итак, по порядку:
Новенькое — широко используемые функции form_for_model и form_for_instance оказывается уже не актуальны. Они работают, но есть более новый и интересный метод получения формы для модели — ModelForm.
В отличии от форм полученных через form_for_model и form_for_instance, формы от ModelForm можно наследовать (а вот это очень интересно в нашем случае!!!).
Поэтому, как только эти два пункта появились в моей голове, я сразу же открыл Python-консоль и начал экспериментировать с кодом. Не уверен на 100%, что мое решение максимально красивое, но тем не менее оно вполне простое и аккуратное (особенно если учесть, что времени на «подумать» особо не было)...
Итак кусочек моего forms.py ниже.
from django.newforms.models import ModelForm
from django.contrib.auth.models import User
from django import newforms as forms
from profile.models import UserProfile
class ProfileForm(ModelForm):
class Meta:
model = UserProfile
fields=('country', 'city', 'post_zip', 'address')
class ProfileFormFull(ProfileForm):
def __init__(self, *args, **kwargs):
super(ProfileFormFull, self).__init__(*args, **kwargs)
self.fields['first_name'] = User._meta.get_field('first_name').formfield()
self.fields['last_name'] = User._meta.get_field('last_name').formfield()
def save_user_data(self):
if self.instance:
user = self.instance.user
user.first_name = self.cleaned_data['first_name']
user.last_name = self.cleaned_data['last_name']
return user.save()
else:
raise instance.DoesNotExist
Теперь мои комментарии к коду (что я делаю):
- Создаю форму ProfileForm из моей модели UserProfile.
- Наследую от ProfileForm новую форму ProfileFormFull, которую буду расширять для возможности редактирования двух моделей (User и UserProfile) в одной форме.
- Суть расширения в добавлении двух полей из модели User (first_name и last_name) и метода save_user_data который эти поля сохраняет в модель User.
- Остальные поля из модели UserProfile обрабатываются автоматически созданным методом save.
- При сохранении формы нужно вызывать два метода — save и save_user_data. Я специально оставил эту возможность для моего случая. Но вы можете сделать сохранение save_user_data внутри save, расширив его.
Как выглядит код, который работает с этой формой:
@login_required
@render_to('profile/main.html')
def profile_save(request):
if request.method == 'GET':
return HttpResponseRedirect(reverse('profile.views.profile_page'))
profile, created = UserProfile.objects.get_or_create(user=request.user)
form = ProfileFormFull(data=request.POST, instance=profile)
if form.is_valid():
form.save()
form.save_user_data()
request.user.message_set.create(message="Your profile details saved.")
return HttpResponseRedirect(reverse('profile.views.profile_page'))
else:
return {'profile': profile, 'profile_form': form, 'user': request.user}
На этом пожалуй откланиваюсь. Был бы очень рад любым конструктивным замечаниям и вопросам в комментариях. Хотелось бы унифицировать кусок кода с формой (forms.py) и может вынести в отдельный сниппет.
PS: обязательно почитайте документацию по ModelForm, интересно :-)
Не изобретая велосипед 5
Хотел поделиться списком django-проектов, которые могут быть очень полезны при старте любого нового проекта. Чтобы, как говориться, «не изобретать велосипед».
Из того, что пробывал лично я, советую:
- sorl-thumbnail – для генерации thumbnail картинок для ImageField, очень удобно, может делать crop, изменение качества для jpg, подробнее смотрите на сайте проекта (все параметры задаются прямо через темплейттаг, так что верстальщики не будут задалбывать каждый раз когда нужно поменять размер картинки на фронтенде);
- django-registration – для автоматизации регистрации юзеров (проверка имейла итд), скорее всего будете затачивать под себя, так что годиться но с напильником;
- django-tagging – неплохая реализация тегирования для джанги, но требует доработки в плане оптимизации запросов к базе;
- django-template-utils – можно выцепить полезный код или использовать «как есть» при работе с темплейтами, полезные темплейттаги;
Из того, что не пробывал, но советую пробывать читателям:
- django-voting – система рейтинга для контента (работает для юбой модели);
- django-evolution – новый проект для тракинга изменений схемы базы (моделей); сам не тестил, но отзывы хорошие;
- django-atompub – для генерации atom-фидов (feeds), намного гибче стандартного джанговского, очень хорошие отзывы;
- django-rest-interface – для создания ReST API, возможно для работы с ajax;
- typogrify – для обработки вводимого/выводимого текста (правильные кавычки, тире итд); актуально только для англоязычных проектов, наша типографика отличается (читайте Лебедева);
- django-localdates – для корректной локализации дат; интересно как работает с русскими датами, кто может протестировать?
Отдельно. Готовые к бою приложения:
- ByteFlow – блогодвижок, разработка моего товарища, оч советую (есть скрипт миграции с wordpress).
- feedjack – приложение, feed агрегатор на джанге. Полезно для коммунити-сайтов.
- webthumb-api – генерация превьюшек реальных сайтов (передаешь ссылку получаешь картинку).
- django-diario – блогодвижок, не смотрел код, кто пробывал или попробует – напишите комментарий о своем впечатлении.
- django-photologue – фотогалерея, тоже сам не пробывал, но судя по описанию довольно «продвинутое» приложение (кто попробывал – отпишитесь).
- snapboard – готовый движок форума, опять же не пробывал, если есть впечатления – напишите.






