# Copyright © Fundacja Nowoczesna Polska. See NOTICE for more information.
#
import datetime
+from functools import wraps
from django.conf import settings
from django.db import models
+from django.db.models.fields.files import FieldFile
from django.db.models import signals
from django import forms
from django.forms.widgets import flatatt
), ], ["^catalogue\.fields\.JSONField"])
except ImportError:
pass
+
+
+class OverwritingFieldFile(FieldFile):
+ """
+ Deletes the old file before saving the new one.
+ """
+
+ def save(self, *args, **kwargs):
+ leave = kwargs.pop('leave', None)
+ if not leave and self:
+ self.delete(save=False)
+ return super(OverwritingFieldFile, self).save(*args, **kwargs)
+
+
+class OverwritingFileField(models.FileField):
+ attr_class = OverwritingFieldFile
+
from newtagging.models import TagBase, tags_updated
from newtagging import managers
-from catalogue.fields import JSONField
+from catalogue.fields import JSONField, OverwritingFileField
from catalogue.utils import ExistingFile
from librarian import dcparser, html, epub, NoDublinCore
class BookMedia(models.Model):
type = models.CharField(_('type'), choices=MEDIA_FORMATS, max_length="100")
name = models.CharField(_('name'), max_length="100")
- file = models.FileField(_('file'), upload_to=book_upload_path())
+ file = OverwritingFileField(_('file'), upload_to=book_upload_path())
uploaded_at = models.DateTimeField(_('creation date'), auto_now_add=True, editable=False)
extra_info = JSONField(_('extra information'), default='{}', editable=False)
book = models.ForeignKey('Book', related_name='media')
def save(self, *args, **kwargs):
try:
- b = BookMedia.objects.get(pk=self.pk)
+ old = BookMedia.objects.get(pk=self.pk)
except BookMedia.DoesNotExist, e:
pass
else:
- # if file is replaced, delete the old one
- if self.file.path != b.file.path:
- b.file.delete(save=False)
# if name changed, change the file name, too
- elif self.name != b.name:
- self.file.save(None, ExistingFile(self.file.path))
+ if self.name != old.name:
+ self.file.save(None, ExistingFile(self.file.path), save=False, leave=True)
super(BookMedia, self).save(*args, **kwargs)
extra_info = self.get_extra_info_value()
# -*- coding: utf-8 -*-
-
+from os.path import basename
from django.core.files.base import ContentFile
from catalogue.test_utils import *
def setUp(self):
WLTestCase.setUp(self)
self.file = ContentFile('X')
+ self.file2 = ContentFile('Y')
+ self.book = models.Book.objects.create(slug='test-book')
def test_diacritics(self):
- bm = models.BookMedia.objects.create(type="ogg",
- name="Zażółć gęślą jaźń")
- bm.file.save(bm.name, self.file)
- self.assertEqual(bm.file.name.rsplit('/', 1)[-1], 'zazolc-gesla-jazn.ogg')
-
+ bm = models.BookMedia(book=self.book, type="ogg",
+ name=u"Zażółć gęślą jaźń")
+ bm.file.save(None, self.file)
+ self.assertEqual(basename(bm.file.name), 'zazolc-gesla-jazn.ogg')
def test_long_name(self):
- bm = models.BookMedia.objects.create(type="ogg",
+ bm = models.BookMedia(book=self.book, type="ogg",
name="Some very very very very very very very very very very very very very very very very long file name")
-
- # save twice so Django adds some stuff
bm.file.save(bm.name, self.file)
- bm.file.save(bm.name, self.file)
- bm.save()
# reload to see what was really saved
bm = models.BookMedia.objects.get(pk=bm.pk)
self.assertEqual(bm.file.size, 1)
+ def test_overwrite(self):
+ """
+ File gets overwritten with same filename on update.
+ """
+
+ bm = models.BookMedia(book=self.book, type='ogg',
+ name="Some media")
+ bm.file.save(None, self.file)
+ bm.file.save(None, self.file2)
+
+ self.assertEqual(bm.file.read(), 'Y')
+ self.assertEqual(basename(bm.file.name), 'some-media.ogg')
+
+ def test_no_clobber(self):
+ """
+ File save doesn't clobber some other media with similar name.
+ """
+
+ bm = models.BookMedia(book=self.book, type='ogg',
+ name="Tytul")
+ bm.file.save(None, self.file)
+ bm2 = models.BookMedia(book=self.book, type='ogg',
+ name="Tytuł")
+ bm2.file.save(None, self.file2)
+ self.assertEqual(basename(bm.file.name), 'tytul.ogg')
+ self.assertNotEqual(basename(bm2.file.name), 'tytul.ogg')
+ self.assertEqual(bm.file.read(), 'X')
+ self.assertEqual(bm2.file.read(), 'Y')
+
+ def test_change_name(self):
+ """
+ File name reflects name change.
+ """
+
+ bm = models.BookMedia(book=self.book, type='ogg', name="Title")
+ bm.file.save(None, self.file)
+ bm.name = "Other Title"
+ bm.save()
+ self.assertEqual(basename(bm.file.name), 'other-title.ogg')
+ self.assertEqual(bm.file.read(), 'X')
+
+ def test_change_name_no_clobber(self):
+ """
+ File name after change won't clobber some other file
+ with similar name.
+ """
+
+ bm = models.BookMedia(book=self.book, type='ogg', name="Title")
+ bm.file.save(None, self.file)
+ bm2 = models.BookMedia(book=self.book, type='ogg', name="Other title")
+ bm2.file.save(None, self.file2)
+ bm2.name = "Title"
+ bm2.save()
+ self.assertNotEqual(basename(bm2.file.name), 'title.ogg')
+ self.assertEqual(bm.file.read(), 'X')
+ self.assertEqual(bm2.file.read(), 'Y')