220 lines
7.7 KiB
220 lines
7.7 KiB
# -*- coding: utf-8 -*-
|
|
# Copyright 2013 XCG Consulting (http://odoo.consulting)
|
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl)
|
|
from cStringIO import StringIO
|
|
import json
|
|
import pkg_resources
|
|
import os
|
|
import sys
|
|
from base64 import b64decode
|
|
import requests
|
|
from tempfile import NamedTemporaryFile
|
|
from odoo import api, _
|
|
from odoo import exceptions
|
|
from odoo.report.report_sxw import report_sxw
|
|
import logging
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
try:
|
|
from py3o.template.helpers import Py3oConvertor
|
|
from py3o.template import Template
|
|
except ImportError:
|
|
logger.debug('Cannot import py3o.template')
|
|
try:
|
|
from py3o.formats import Formats
|
|
except ImportError:
|
|
logger.debug('Cannot import py3o.formats')
|
|
|
|
|
|
_extender_functions = {}
|
|
|
|
|
|
class TemplateNotFound(Exception):
|
|
pass
|
|
|
|
|
|
def py3o_report_extender(report_xml_id=None):
|
|
"""
|
|
A decorator to define function to extend the context sent to a template.
|
|
This will be called at the creation of the report.
|
|
The following arguments will be passed to it:
|
|
- ir_report: report instance
|
|
- localcontext: The context that will be passed to the report engine
|
|
If no report_xml_id is given the extender is registered for all py3o
|
|
reports
|
|
Idea copied from CampToCamp report_webkit module.
|
|
|
|
:param report_xml_id: xml id of the report
|
|
:return: a decorated class
|
|
"""
|
|
global _extender_functions
|
|
|
|
def fct1(fct):
|
|
_extender_functions.setdefault(report_xml_id, []).append(fct)
|
|
return fct
|
|
return fct1
|
|
|
|
|
|
@py3o_report_extender()
|
|
def defautl_extend(report_xml, localcontext):
|
|
# add the base64decode function to be able do decode binary fields into
|
|
# the template
|
|
localcontext['b64decode'] = b64decode
|
|
localcontext['report_xml'] = report_xml
|
|
|
|
|
|
class Py3oParser(report_sxw):
|
|
"""Custom class that use Py3o to render libroffice reports.
|
|
Code partially taken from CampToCamp's webkit_report."""
|
|
|
|
def get_template(self, report_obj):
|
|
"""private helper to fetch the template data either from the database
|
|
or from the default template file provided by the implementer.
|
|
|
|
ATM this method takes a report definition recordset
|
|
to try and fetch the report template from database. If not found it
|
|
will fallback to the template file referenced in the report definition.
|
|
|
|
@param report_obj: a recordset representing the report defintion
|
|
@type report_obj: odoo.model.recordset instance
|
|
|
|
@returns: string or buffer containing the template data
|
|
|
|
@raises: TemplateNotFound which is a subclass of
|
|
odoo.exceptions.DeferredException
|
|
"""
|
|
|
|
tmpl_data = None
|
|
|
|
if report_obj.py3o_template_id and report_obj.py3o_template_id.id:
|
|
# if a user gave a report template
|
|
tmpl_data = b64decode(
|
|
report_obj.py3o_template_id.py3o_template_data
|
|
)
|
|
|
|
elif report_obj.py3o_template_fallback:
|
|
tmpl_name = report_obj.py3o_template_fallback
|
|
flbk_filename = None
|
|
if report_obj.module:
|
|
# if the default is defined
|
|
flbk_filename = pkg_resources.resource_filename(
|
|
"odoo.addons.%s" % report_obj.module,
|
|
tmpl_name,
|
|
)
|
|
elif os.path.isabs(tmpl_name):
|
|
# It is an absolute path
|
|
flbk_filename = os.path.normcase(os.path.normpath(tmpl_name))
|
|
if flbk_filename and os.path.exists(flbk_filename):
|
|
# and it exists on the fileystem
|
|
with open(flbk_filename, 'r') as tmpl:
|
|
tmpl_data = tmpl.read()
|
|
|
|
if tmpl_data is None:
|
|
# if for any reason the template is not found
|
|
raise TemplateNotFound(
|
|
_(u'No template found. Aborting.'),
|
|
sys.exc_info(),
|
|
)
|
|
|
|
return tmpl_data
|
|
|
|
def _extend_parser_context(self, parser_instance, report_xml):
|
|
# add default extenders
|
|
for fct in _extender_functions.get(None, []):
|
|
fct(report_xml, parser_instance.localcontext)
|
|
# add extenders for registered on the template
|
|
xml_id = report_xml.get_external_id().get(report_xml.id)
|
|
if xml_id in _extender_functions:
|
|
for fct in _extender_functions[xml_id]:
|
|
fct(report_xml, parser_instance.localcontext)
|
|
|
|
def create_single_pdf(self, cr, uid, ids, data, report_xml, context=None):
|
|
""" Overide this function to generate our py3o report
|
|
"""
|
|
if report_xml.report_type != 'py3o':
|
|
return super(Py3oParser, self).create_single_pdf(
|
|
cr, uid, ids, data, report_xml, context=context
|
|
)
|
|
|
|
parser_instance = self.parser(cr, uid, self.name2, context=context)
|
|
parser_instance.set_context(
|
|
self.getObjects(cr, uid, ids, context),
|
|
data, ids, report_xml.report_type
|
|
)
|
|
self._extend_parser_context(parser_instance, report_xml)
|
|
|
|
tmpl_data = self.get_template(report_xml)
|
|
|
|
in_stream = StringIO(tmpl_data)
|
|
out_stream = StringIO()
|
|
template = Template(in_stream, out_stream)
|
|
localcontext = parser_instance.localcontext
|
|
if report_xml.py3o_is_local_fusion:
|
|
template.render(localcontext)
|
|
in_stream = out_stream
|
|
datadict = {}
|
|
else:
|
|
expressions = template.get_all_user_python_expression()
|
|
py_expression = template.convert_py3o_to_python_ast(expressions)
|
|
convertor = Py3oConvertor()
|
|
data_struct = convertor(py_expression)
|
|
datadict = data_struct.render(localcontext)
|
|
|
|
filetype = report_xml.py3o_filetype
|
|
is_native = Formats().get_format(filetype).native
|
|
if is_native:
|
|
res = out_stream.getvalue()
|
|
else: # Call py3o.server to render the template in the desired format
|
|
in_stream.seek(0)
|
|
files = {
|
|
'tmpl_file': in_stream,
|
|
}
|
|
fields = {
|
|
"targetformat": filetype,
|
|
"datadict": json.dumps(datadict),
|
|
"image_mapping": "{}",
|
|
}
|
|
if report_xml.py3o_is_local_fusion:
|
|
fields['skipfusion'] = '1'
|
|
r = requests.post(
|
|
report_xml.py3o_server_id.url, data=fields, files=files)
|
|
if r.status_code != 200:
|
|
# server says we have an issue... let's tell that to enduser
|
|
raise exceptions.Warning(
|
|
_('Fusion server error %s') % r.text,
|
|
)
|
|
|
|
# Here is a little joke about Odoo
|
|
# we do nice chunked reading from the network...
|
|
chunk_size = 1024
|
|
with NamedTemporaryFile(
|
|
suffix=filetype,
|
|
prefix='py3o-template-'
|
|
) as fd:
|
|
for chunk in r.iter_content(chunk_size):
|
|
fd.write(chunk)
|
|
fd.seek(0)
|
|
# ... but odoo wants the whole data in memory anyways :)
|
|
res = fd.read()
|
|
|
|
return res, filetype
|
|
|
|
def create(self, cr, uid, ids, data, context=None):
|
|
""" Override this function to handle our py3o report
|
|
"""
|
|
env = api.Environment(cr, uid, context)
|
|
report_xmls = env['ir.actions.report.xml'].search(
|
|
[('report_name', '=', self.name[7:])])
|
|
if not report_xmls:
|
|
return super(Py3oParser, self).create(
|
|
cr, uid, ids, data, context=context
|
|
)
|
|
|
|
result = self.create_source_pdf(
|
|
cr, uid, ids, data, report_xmls[0], context
|
|
)
|
|
|
|
if not result:
|
|
return False, False
|
|
return result
|