# Copyright © Fundacja Nowoczesna Polska. See NOTICE for more information.
#
from django.apps import apps
+from django.core.files.base import ContentFile
from django.contrib.sites.models import Site
from django.db import connection, models, transaction
from django.template.loader import render_to_string
from django.urls import reverse
-from django.utils.translation import ugettext_lazy as _
+from django.utils.translation import gettext_lazy as _
from django.conf import settings
from slugify import slugify
-
+from librarian.cover import make_cover
+from librarian.dcparser import BookInfo
import apiclient
from documents.helpers import cached_in_field, GalleryMerger
import os
import shutil
import re
+from urllib.parse import urljoin
+
class Book(models.Model):
""" A document edited on the wiki """
dc_cover_image = models.ForeignKey(Image, blank=True, null=True,
db_index=True, on_delete=models.SET_NULL, editable=False)
dc = models.JSONField(null=True, editable=False)
+ cover = models.FileField(blank=True, upload_to='documents/cover')
+
+ dc_slug = models.CharField(
+ max_length=2048,
+ null=True, blank=True,
+ editable=False,
+ )
catalogue_book = models.ForeignKey(
'catalogue.Book',
- models.DO_NOTHING,
- to_field='slug',
+ models.PROTECT,
null=True, blank=True,
- db_constraint=False,
- editable=False, db_index=True,
+ editable=False,
related_name='document_books',
related_query_name='document_book',
)
try:
changes = self.get_current_changes(publishable=True)
except self.NoTextError:
- raise AssertionError(_('Not all chunks have publishable revisions.'))
+ raise AssertionError(_('Not all chunks have approved revisions.'))
from librarian import NoDublinCore, ParseError, ValidationError
def refresh_dc_cache(self):
update = {
- 'catalogue_book_id': None,
+ 'dc_slug': None,
'dc_cover_image': None,
}
info = self.book_info()
if info is not None:
- update['catalogue_book_id'] = info.url.slug
+ update['dc_slug'] = info.url.slug
if info.cover_source:
try:
image = Image.objects.get(pk=int(info.cover_source.rstrip('/').rsplit('/', 1)[-1]))
except:
pass
else:
- if info.cover_source == image.get_full_url():
+ if info.cover_source.rstrip('/') == image.get_full_url().rstrip('/'):
update['dc_cover_image'] = image
update['dc'] = info.to_dict()
Book.objects.filter(pk=self.pk).update(**update)
}
Book.objects.filter(pk=self.pk).update(**update)
self.refresh_dc_cache()
+ self.build_cover()
+
+ def build_cover(self):
+ width, height = 212, 300
+ try:
+ xml = self.materialize(publishable=True).encode('utf-8')
+ info = BookInfo.from_bytes(xml)
+ kwargs = {}
+ if self.project is not None:
+ if self.project.logo_mono or self.project.logo:
+ kwargs['cover_logo'] = (self.project.logo_mono or self.project.logo).path
+ cover = make_cover(info, width=width, height=height, **kwargs)
+ out = BytesIO()
+ ext = cover.ext()
+ cover.save(out)
+ self.cover.save(f'{self.slug}.{ext}', out, save=False)
+ type(self).objects.filter(pk=self.pk).update(cover=self.cover)
+ except:
+ type(self).objects.filter(pk=self.pk).update(cover='')
# Materializing & publishing
# ==========================
if not fake:
book_xml = self.materialize(changes=changes)
data = {"book_xml": book_xml, "days": days, "hidden": hidden}
+ if self.project is not None:
+ if self.project.logo:
+ data['logo'] = urljoin(
+ 'https://' + Site.objects.get_current().domain,
+ self.project.logo.url,
+ )
+ if self.project.logo_mono:
+ data['logo_mono'] = urljoin(
+ 'https://' + Site.objects.get_current().domain,
+ self.project.logo_mono.url,
+ )
+ if self.project.logo_alt:
+ data['logo_alt'] = self.project.logo_alt
if host:
data['gallery_url'] = host + self.gallery_url()
apiclient.api_call(user, "books/", data, beta=beta)