Source code for org.pdf.ticket

from __future__ import annotations

from onegov.org import _
from datetime import date
from functools import partial
from io import BytesIO, StringIO

from bleach import Cleaner
from bleach.linkifier import LinkifyFilter
from lxml import etree
from pdfdocument.document import MarkupParagraph
from reportlab.platypus import Paragraph

from onegov.chat import MessageCollection
from onegov.org.constants import TICKET_STATES, PAYMENT_STATES, PAYMENT_SOURCES
from onegov.org.layout import TicketLayout
from onegov.org.models.ticket import ticket_submitter
from onegov.org.views.message import view_messages_feed
from onegov.pdf import Pdf, page_fn_header
from onegov.qrcode import QrCode
from html5lib.filters.whitespace import Filter as WhitespaceFilter
from reportlab.lib import colors
from reportlab.lib.units import cm
from reportlab.lib.utils import ImageReader


from typing import overload, Any, Literal, TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Callable, Iterable, Mapping, Sequence
    from bleach.callbacks import _HTMLAttrs
    from bleach.sanitizer import _Filter
    from gettext import GNUTranslations
    from onegov.org.layout import DefaultLayout
    from onegov.org.request import OrgRequest
    from onegov.pdf.templates import Template
    from onegov.ticket import Ticket
    from reportlab.lib.styles import PropertySet
    from reportlab.pdfgen.canvas import Canvas
    from reportlab.platypus.tables import _TableCommand, TableStyle


[docs] class TicketQrCode(QrCode):
[docs] _border = 0
[docs] class TicketPdf(Pdf): def __init__(self, *args: Any, **kwargs: Any) -> None: locale = kwargs.pop('locale', None) translations = kwargs.pop('translations', None) ticket = kwargs.pop('ticket') layout = kwargs.pop('layout') qr_payload = kwargs.pop('qr_payload') super().__init__(*args, **kwargs)
[docs] self.ticket: Ticket = ticket
[docs] self.locale: str | None = locale
[docs] self.translations: dict[str, GNUTranslations] = translations
[docs] self.layout: DefaultLayout = layout
# Modification for the footer left on all pages self.doc.author = self.translate(_('Source')) + f': {self.doc.author}' self.doc.qr_payload = qr_payload # type:ignore[attr-defined]
[docs] def translate(self, text: str) -> str: """ Translates the given string. """ if not hasattr(text, 'interpolate'): return text translator = ( self.translations.get(self.locale) if self.locale else None ) translated = translator.gettext(text) if translator else text return text.interpolate(translated)
[docs] def h1(self, title: str) -> None: # type:ignore[override] """ Translated H1. """ super().h1(self.translate(title))
[docs] def h2(self, title: str) -> None: # type:ignore[override] """ Translated H2. """ super().h2(self.translate(title))
[docs] def h3(self, title: str) -> None: # type:ignore[override] """ Translated H3. """ super().h3(self.translate(title))
@staticmethod @staticmethod @property
[docs] def page_fn(self) -> Callable[[Canvas, Template], None]: """ First page the same as later except Qr-Code ...""" return self.page_fn_header_and_footer_qr
@property
[docs] def page_fn_later(self) -> Callable[[Canvas, Template], None]: return self.page_fn_header_and_footer
@overload # type:ignore[override]
[docs] def table( self, data: Sequence[Sequence[str | Paragraph]], columns: Literal['even'] | Sequence[float | None] | None, style: TableStyle | Iterable[_TableCommand] | None = None, ratios: Literal[False] = False, border: bool = True, first_bold: bool = True ) -> None: ...
@overload def table( self, data: Sequence[Sequence[str | Paragraph]], columns: Literal['even'] | list[float] | None, style: TableStyle | Iterable[_TableCommand] | None = None, *, ratios: Literal[True], border: bool = True, first_bold: bool = True ) -> None: ... @overload def table( self, data: Sequence[Sequence[str | Paragraph]], columns: Literal['even'] | list[float] | None, style: TableStyle | Iterable[_TableCommand] | None, ratios: Literal[True], border: bool = True, first_bold: bool = True ) -> None: ... @overload def table( self, data: Sequence[Sequence[str | Paragraph]], columns: Literal['even'] | Sequence[float | None] | None, style: TableStyle | Iterable[_TableCommand] | None = None, ratios: bool = False, border: bool = True, first_bold: bool = True ) -> None: ... def table( self, data: Sequence[Sequence[str | Paragraph]], columns: Literal['even'] | Sequence[float | None] | None, style: TableStyle | Iterable[_TableCommand] | None = None, ratios: bool = False, border: bool = True, first_bold: bool = True ) -> None: if border: # FIXME: What if we want to pass a style in? style = list(self.style.table) style.append(('LINEBELOW', (0, 0), (-1, -1), 0.25, colors.black)) if not first_bold: return super().table(data, columns, style, ratios) for row in data: if isinstance(row[0], Paragraph): continue row[0] = MarkupParagraph( # type:ignore[index] self.translate(row[0]), style=self.style.bold) return super().table(data, columns, style, ratios)
[docs] def ticket_summary(self, html: str | None, linkify: bool = True) -> None: """A copy of the mini_html adapted for ticket summary. We have to guarantee some backwards compatibility here whenever we change the templates and thereby the snapshot of tickets. Must work for templates: - directory_entry_submission - display_event - display_form - reservations """ if not html or html == '<p></p>': return # Remove unwanted markup tags = ['p', 'br', 'strong', 'b', 'em', 'li', 'ol', 'ul', 'li'] ticket_summary_tags = ['dl', 'dt', 'dd', 'h2'] tags += ticket_summary_tags attributes: dict[str, list[str]] = {} filters: list[_Filter] = [WhitespaceFilter] if linkify: link_color = self.link_color underline_links = self.underline_links underline_width = self.underline_width def colorize( attrs: _HTMLAttrs, new: bool = False ) -> _HTMLAttrs: # phone numbers appear here but are escaped, skip... if not attrs.get((None, 'href')): # FIXME: The bleach stubs appear to be incorrect # since this definitely works at runtime # but we may be able to return an empty # dictionary or attrs instead of None return None # type:ignore[return-value] attrs[(None, 'color')] = link_color if underline_links: attrs[(None, 'underline')] = '1' attrs[('a', 'underlineColor')] = link_color attrs[('a', 'underlineWidth')] = underline_width return attrs tags.append('a') attributes['a'] = ['href'] filters.append( partial( LinkifyFilter, parse_email=True, callbacks=[colorize]) ) cleaner = Cleaner( tags=tags, attributes=attributes, strip=True, filters=filters ) html = cleaner.clean(html) # Todo: phone numbers with href="tel:.." are cleaned out tree = etree.parse(StringIO(html), etree.HTMLParser()) body_element = tree.find('body') assert body_element is not None for element in body_element: if element.tag == 'dl': data: list[list[Paragraph | str]] = [] for item in element: if item.tag == 'dt': p = MarkupParagraph( self.inner_html(item), self.style.bold) data.append([p, '']) if item.tag == 'dd': data[-1][1] = MarkupParagraph(self.inner_html(item)) if data: self.table(data, 'even') elif element.tag == 'h2': # Fieldset titles self.h2(self.inner_html(element)) elif element.tag == 'ul': items = [ [MarkupParagraph(self.inner_html(item))] for item in element ] self.table(items, 'even')
[docs] def ticket_metadata(self) -> None: layout = self.layout handler = self.ticket.handler group = handler.group or self.ticket.group created_dt = layout.to_timezone(self.ticket.created, layout.timezone) created = layout.format_date(created_dt, 'datetime') # Ticket meta info if hasattr(self.ticket, 'reference_group'): subject = self.ticket.reference_group(layout.request) else: subject = self.ticket.title submitter = ticket_submitter(self.ticket) ticket_state = self.translate(TICKET_STATES[self.ticket.state]) owner = self.ticket.user.username if self.ticket.user else '' def seconds(time: float | None) -> str: return self.layout.format_seconds(time) if time else '' meta_fields = { 'submitter_name': _('Name'), 'submitter_address': _('Address'), 'submitter_phone': _('Phone') } # pep572 still a cool thing submitter_meta = [ [text, layout.linkify(value)] for field, text in meta_fields.items() if (value := getattr(handler, field)) ] data = [ [_('Subject'), subject], [_('Submitter'), submitter], *submitter_meta, [_('State'), ticket_state], [_('Group'), group], [_('Owner'), owner], [_('Created'), created], [_('Reaction Time'), seconds(self.ticket.reaction_time)], [_('Process Time'), seconds(self.ticket.process_time)], ] # In case of imported events.. event_source = handler.data.get('source') if event_source and self.layout.request.is_manager: data.append([self.translate(_('Event Source')), event_source]) self.table(data, 'even')
[docs] def ticket_payment(self) -> None: price = self.ticket.handler.payment if not price: return state = self.translate(PAYMENT_STATES[price.state]) # credit card self.h2(_('Payment')) amount = f'{self.layout.format_number(price.net_amount)}' self.table([ [_('Total Amount'), f'{amount} {price.currency}'], [_('State'), state], [_('Source'), self.translate(PAYMENT_SOURCES[price.source])], [_('Fee'), f'{self.layout.format_number(price.fee)}'], ], 'even')
[docs] def p_markup( self, text: str, style: PropertySet | None = None ) -> None: super().p_markup(self.translate(text), style)
[docs] def ticket_timeline(self, msg_feed: Mapping[str, Any] | None) -> None: """Will parse the timeline from view_messages_feed """ if not msg_feed or not msg_feed['messages']: return tables: dict[str, list[list[str | None]]] = {} for msg in reversed(msg_feed['messages']): table = tables.setdefault(msg['date'], []) row = self.extract_feed_info(msg['html']) if row is None: continue table.append(row) for date_, data in tables.items(): self.h4(date_) # FIXME: Why does `None` work? is it because of `MarkupParagraph` # internals? Why not use an empty string? Does that do # something different? self.table(data, 'even', first_bold=False) # type:ignore
@staticmethod
[docs] def extract_feed_info(html: str) -> list[str | None] | None: """ Must be able to parse templates message_{message.type}.pt and return the useful data in cleaned form. """ if not html or html == '<p></p>': return None # Remove unwanted markup tags = ['p', 'br', 'strong', 'b', 'em', 'li', 'ol', 'ul', 'li'] ticket_summary_tags = ['dl', 'dt', 'dd', 'h2', 'div'] tags += ticket_summary_tags attributes = {} filters = [WhitespaceFilter] attributes['div'] = ['class'] cleaner = Cleaner( tags=tags, attributes=attributes, strip=True, filters=filters ) html = cleaner.clean(html) tree = etree.parse(StringIO(html), etree.HTMLParser()) data = [] body_element = tree.find('body') assert body_element is not None for el in body_element: if el.tag == 'div': class_ = el.attrib['class'] if class_ == 'timestamp': data = [TicketPdf.inner_html(el), None] if class_ == 'text': data[1] = TicketPdf.inner_html(el) return data
@classmethod
[docs] def from_ticket( cls, request: OrgRequest, ticket: Ticket ) -> BytesIO: """ Creates a PDF representation of the ticket. It is sensible to the templates used to render the message feed and the summary of the ticket coming from ticket handler. With this approach, snapshotted summaries are supported. """ result = BytesIO() handler = ticket.handler layout = TicketLayout(ticket, request) pdf = cls( result, title=ticket.number, created=f'{date.today():%d.%m.%Y}', link_color='#00538c', underline_links=True, author=request.host_url, ticket=ticket, translations=request.app.translations, locale=request.locale, layout=layout, qr_payload=request.link( ticket, name=request.is_manager and None or 'status') ) pdf.init_a4_portrait( page_fn=pdf.page_fn, page_fn_later=pdf.page_fn_later ) pdf.h(f'{request.translate(_("Ticket"))} {ticket.number}') pdf.spacer() deleted_message = None if handler.deleted: summary = ticket.snapshot.get('summary') deleted_message = _('The record behind this ticket was removed. ' 'The following information is a snapshot ' 'kept for future reference.') else: summary = handler.get_summary(request) pdf.ticket_metadata() pdf.h1(_('Summary')) if deleted_message: pdf.p(pdf.translate(deleted_message)) pdf.spacer() pdf.ticket_summary(summary) pdf.ticket_payment() # If used for the user instead of the manager... messages = MessageCollection( request.session, channel_id=ticket.number ) if not request.is_manager: messages.type = request.app.settings.org.public_ticket_messages pdf.h1(request.translate(_('Timeline'))) pdf.ticket_timeline(view_messages_feed(messages, request)) pdf.generate() result.seek(0) return result