2018-08-22 21:18:12 +00:00
|
|
|
# -*- mode: python; coding: utf-8 -*-
|
2017-01-15 23:01:18 +00:00
|
|
|
# Re2o est un logiciel d'administration développé initiallement au rezometz. Il
|
|
|
|
# se veut agnostique au réseau considéré, de manière à être installable en
|
|
|
|
# quelques clics.
|
|
|
|
#
|
|
|
|
# Copyright © 2017 Gabriel Détraz
|
2019-09-29 14:02:28 +00:00
|
|
|
# Copyright © 2017 Lara Kermarec
|
2017-01-15 23:01:18 +00:00
|
|
|
# Copyright © 2017 Augustin Lemesle
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License along
|
|
|
|
# with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
2018-04-09 17:40:46 +00:00
|
|
|
"""tex.py
|
|
|
|
Module in charge of rendering some LaTex templates.
|
|
|
|
Used to generated PDF invoice.
|
|
|
|
"""
|
2017-01-15 23:01:18 +00:00
|
|
|
|
2018-08-24 15:00:02 +00:00
|
|
|
|
2018-04-14 13:39:51 +00:00
|
|
|
import tempfile
|
|
|
|
from subprocess import Popen, PIPE
|
|
|
|
import os
|
|
|
|
from datetime import datetime
|
|
|
|
|
2019-01-03 18:52:06 +00:00
|
|
|
from django.db import models
|
2016-07-09 15:16:44 +00:00
|
|
|
from django.template.loader import get_template
|
2017-11-14 19:00:24 +00:00
|
|
|
from django.http import HttpResponse
|
2016-07-09 15:16:44 +00:00
|
|
|
from django.conf import settings
|
2017-11-14 19:00:24 +00:00
|
|
|
from django.utils.text import slugify
|
2019-01-03 18:52:06 +00:00
|
|
|
from django.utils.translation import ugettext_lazy as _
|
|
|
|
|
|
|
|
from re2o.mixins import AclMixin, RevMixin
|
2019-01-03 21:34:45 +00:00
|
|
|
from preferences.models import CotisationsOption
|
2016-07-09 15:16:44 +00:00
|
|
|
|
|
|
|
|
|
|
|
TEMP_PREFIX = getattr(settings, 'TEX_TEMP_PREFIX', 'render_tex-')
|
|
|
|
CACHE_PREFIX = getattr(settings, 'TEX_CACHE_PREFIX', 'render-tex')
|
|
|
|
CACHE_TIMEOUT = getattr(settings, 'TEX_CACHE_TIMEOUT', 86400) # 1 day
|
|
|
|
|
|
|
|
|
2018-04-14 13:39:51 +00:00
|
|
|
def render_invoice(_request, ctx={}):
|
2018-04-09 17:40:46 +00:00
|
|
|
"""
|
|
|
|
Render an invoice using some available information such as the current
|
|
|
|
date, the user, the articles, the prices, ...
|
|
|
|
"""
|
2019-01-03 21:34:45 +00:00
|
|
|
options, _ = CotisationsOption.objects.get_or_create()
|
2018-12-31 22:58:37 +00:00
|
|
|
is_estimate = ctx.get('is_estimate', False)
|
2017-11-14 19:00:24 +00:00
|
|
|
filename = '_'.join([
|
2018-12-31 22:58:37 +00:00
|
|
|
'cost_estimate' if is_estimate else 'invoice',
|
2018-04-14 13:39:51 +00:00
|
|
|
slugify(ctx.get('asso_name', "")),
|
|
|
|
slugify(ctx.get('recipient_name', "")),
|
|
|
|
str(ctx.get('DATE', datetime.now()).year),
|
|
|
|
str(ctx.get('DATE', datetime.now()).month),
|
|
|
|
str(ctx.get('DATE', datetime.now()).day),
|
2017-11-14 19:00:24 +00:00
|
|
|
])
|
2019-01-03 21:34:45 +00:00
|
|
|
templatename = options.invoice_template.template.name.split('/')[-1]
|
|
|
|
r = render_tex(_request, templatename, ctx)
|
2018-04-13 18:58:29 +00:00
|
|
|
r['Content-Disposition'] = 'attachment; filename="{name}.pdf"'.format(
|
|
|
|
name=filename
|
|
|
|
)
|
2017-11-14 19:00:24 +00:00
|
|
|
return r
|
|
|
|
|
2018-04-13 18:58:29 +00:00
|
|
|
|
2019-01-05 18:45:21 +00:00
|
|
|
def render_voucher(_request, ctx={}):
|
|
|
|
"""
|
|
|
|
Render a subscribtion voucher.
|
|
|
|
"""
|
|
|
|
options, _ = CotisationsOption.objects.get_or_create()
|
|
|
|
filename = '_'.join([
|
|
|
|
'voucher',
|
|
|
|
slugify(ctx.get('asso_name', "")),
|
2019-01-10 23:39:16 +00:00
|
|
|
slugify(ctx.get('firstname', "")),
|
|
|
|
slugify(ctx.get('lastname', "")),
|
|
|
|
str(ctx.get('date_begin', datetime.now()).year),
|
|
|
|
str(ctx.get('date_begin', datetime.now()).month),
|
|
|
|
str(ctx.get('date_begin', datetime.now()).day),
|
2019-01-05 18:45:21 +00:00
|
|
|
])
|
|
|
|
templatename = options.voucher_template.template.name.split('/')[-1]
|
2019-01-10 23:39:16 +00:00
|
|
|
r = render_tex(_request, templatename, ctx)
|
2019-01-05 18:45:21 +00:00
|
|
|
r['Content-Disposition'] = 'attachment; filename="{name}.pdf"'.format(
|
|
|
|
name=filename
|
|
|
|
)
|
|
|
|
return r
|
|
|
|
|
|
|
|
|
2018-07-18 00:08:03 +00:00
|
|
|
def create_pdf(template, ctx={}):
|
2018-08-15 21:56:34 +00:00
|
|
|
"""Creates and returns a PDF from a LaTeX template using pdflatex.
|
|
|
|
|
|
|
|
It create a temporary file for the PDF then read it to return its content.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
template: Path to the LaTeX template.
|
|
|
|
ctx: Dict with the context for rendering the template.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
The content of the temporary PDF file generated.
|
2018-04-09 17:40:46 +00:00
|
|
|
"""
|
2019-09-01 21:25:41 +00:00
|
|
|
context = ctx
|
2017-11-14 19:00:24 +00:00
|
|
|
template = get_template(template)
|
2016-07-09 21:26:17 +00:00
|
|
|
rendered_tpl = template.render(context).encode('utf-8')
|
2018-04-13 18:58:29 +00:00
|
|
|
|
2016-07-09 21:26:17 +00:00
|
|
|
with tempfile.TemporaryDirectory() as tempdir:
|
2018-08-15 21:56:34 +00:00
|
|
|
for _ in range(2):
|
2019-01-10 23:39:16 +00:00
|
|
|
with open("/var/www/re2o/out.log", "w") as f:
|
|
|
|
process = Popen(
|
|
|
|
['pdflatex', '-output-directory', tempdir],
|
|
|
|
stdin=PIPE,
|
|
|
|
stdout=f,#PIPE,
|
|
|
|
)
|
|
|
|
process.communicate(rendered_tpl)
|
2016-07-09 21:26:17 +00:00
|
|
|
with open(os.path.join(tempdir, 'texput.pdf'), 'rb') as f:
|
|
|
|
pdf = f.read()
|
2018-07-18 00:08:03 +00:00
|
|
|
|
|
|
|
return pdf
|
|
|
|
|
|
|
|
|
2018-12-29 13:01:22 +00:00
|
|
|
def escape_chars(string):
|
|
|
|
"""Escape the '%' and the '€' signs to avoid messing with LaTeX"""
|
|
|
|
if not isinstance(string, str):
|
|
|
|
return string
|
|
|
|
mapping = (
|
|
|
|
('€', r'\euro'),
|
|
|
|
('%', r'\%'),
|
|
|
|
)
|
|
|
|
r = str(string)
|
|
|
|
for k, v in mapping:
|
|
|
|
r = r.replace(k, v)
|
|
|
|
return r
|
|
|
|
|
|
|
|
|
2018-07-18 00:08:03 +00:00
|
|
|
def render_tex(_request, template, ctx={}):
|
2018-08-15 21:56:34 +00:00
|
|
|
"""Creates a PDF from a LaTex templates using pdflatex.
|
|
|
|
|
|
|
|
Calls `create_pdf` and send back an HTTP response for
|
2018-07-18 00:08:03 +00:00
|
|
|
accessing this file.
|
2018-08-15 21:56:34 +00:00
|
|
|
|
|
|
|
Args:
|
|
|
|
_request: Unused, but allow using this function as a Django view.
|
|
|
|
template: Path to the LaTeX template.
|
|
|
|
ctx: Dict with the context for rendering the template.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
An HttpResponse with type `application/pdf` containing the PDF file.
|
2018-07-18 00:08:03 +00:00
|
|
|
"""
|
2018-08-15 22:22:07 +00:00
|
|
|
pdf = create_pdf(template, ctx)
|
2016-07-09 21:26:17 +00:00
|
|
|
r = HttpResponse(content_type='application/pdf')
|
|
|
|
r.write(pdf)
|
|
|
|
return r
|