Skip to content

Commit 651813a

Browse files
committed
[MIG] report_py3o to v16
1 parent aa228f0 commit 651813a

File tree

11 files changed

+73
-39
lines changed

11 files changed

+73
-39
lines changed

report_py3o/__manifest__.py

Lines changed: 2 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -4,10 +4,10 @@
44
"name": "Py3o Report Engine",
55
"summary": "Reporting engine based on Libreoffice (ODT -> ODT, "
66
"ODT -> PDF, ODT -> DOC, ODT -> DOCX, ODS -> ODS, etc.)",
7-
"version": "15.0.1.0.0",
7+
"version": "16.0.1.0.0",
88
"category": "Reporting",
99
"license": "AGPL-3",
10-
"author": "XCG Consulting," "ACSONE SA/NV," "Odoo Community Association (OCA)",
10+
"author": "XCG Consulting, ACSONE SA/NV, Odoo Community Association (OCA)",
1111
"website": "https://github.com/OCA/reporting-engine",
1212
"depends": ["web"],
1313
"external_dependencies": {
@@ -21,7 +21,6 @@
2121
},
2222
"data": [
2323
"security/ir.model.access.csv",
24-
"views/menu.xml",
2524
"views/py3o_template.xml",
2625
"views/ir_actions_report.xml",
2726
"demo/report_py3o.xml",

report_py3o/controllers/main.py

Lines changed: 8 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -6,18 +6,17 @@
66
from werkzeug import exceptions
77
from werkzeug.urls import url_decode
88

9-
from odoo.http import request, route
9+
from odoo.http import content_disposition, request, route, serialize_exception
1010
from odoo.tools import html_escape
1111

12-
from odoo.addons.web.controllers import main
13-
from odoo.addons.web.controllers.main import _serialize_exception, content_disposition
12+
from odoo.addons.web.controllers.report import ReportController
1413

1514

16-
class ReportController(main.ReportController):
15+
class ReportController(ReportController):
1716
@route()
1817
def report_routes(self, reportname, docids=None, converter=None, **data):
1918
if converter != "py3o":
20-
return super(ReportController, self).report_routes(
19+
return super().report_routes(
2120
reportname=reportname, docids=docids, converter=converter, **data
2221
)
2322
context = dict(request.env.context)
@@ -44,7 +43,7 @@ def report_routes(self, reportname, docids=None, converter=None, **data):
4443
description="Py3o action report not found for report_name "
4544
"%s" % reportname
4645
)
47-
res, filetype = action_py3o_report._render(docids, data)
46+
res, filetype = ir_action._render(reportname, docids, data)
4847
filename = action_py3o_report.gen_report_download_filename(docids, data)
4948
if not filename.endswith(filetype):
5049
filename = "{}.{}".format(filename, filetype)
@@ -57,7 +56,7 @@ def report_routes(self, reportname, docids=None, converter=None, **data):
5756
return request.make_response(res, headers=http_headers)
5857

5958
@route()
60-
def report_download(self, data, context=None):
59+
def report_download(self, data, context=None, token=None):
6160
"""This function is used by 'qwebactionmanager.js' in order to trigger
6261
the download of a py3o/controller report.
6362
@@ -68,7 +67,7 @@ def report_download(self, data, context=None):
6867
requestcontent = json.loads(data)
6968
url, report_type = requestcontent[0], requestcontent[1]
7069
if "py3o" not in report_type:
71-
return super(ReportController, self).report_download(data, context)
70+
return super().report_download(data, context=context, token=token)
7271
try:
7372
reportname = url.split("/report/py3o/")[1].split("?")[0]
7473
docids = None
@@ -90,6 +89,6 @@ def report_download(self, data, context=None):
9089
response.set_cookie("fileToken", context)
9190
return response
9291
except Exception as e:
93-
se = _serialize_exception(e)
92+
se = serialize_exception(e)
9493
error = {"code": 200, "message": "Odoo Server Error", "data": se}
9594
return request.make_response(html_escape(json.dumps(error)))

report_py3o/demo/report_py3o.xml

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,9 @@
1111
<field name="py3o_filetype">odt</field>
1212
<field name="module">report_py3o</field>
1313
<field name="py3o_template_fallback">demo/res_user.odt</field>
14+
<field
15+
name="print_report_name"
16+
>object.name.replace(' ', '_') + '-demo.odt'</field>
1417
<field name="binding_model_id" ref="base.model_res_users" />
1518
<field name="binding_type">report</field>
1619
</record>

report_py3o/models/ir_actions_report.py

Lines changed: 6 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -157,16 +157,17 @@ def get_from_report_name(self, report_name, report_type):
157157
[("report_name", "=", report_name), ("report_type", "=", report_type)]
158158
)
159159

160-
def _render_py3o(self, res_ids, data):
161-
self.ensure_one()
162-
if self.report_type != "py3o":
160+
@api.model
161+
def _render_py3o(self, report_ref, res_ids, data=None):
162+
report = self._get_report(report_ref)
163+
if report.report_type != "py3o":
163164
raise RuntimeError(
164165
"py3o rendition is only available on py3o report.\n"
165-
"(current: '{}', expected 'py3o'".format(self.report_type)
166+
"(current: '{}', expected 'py3o'".format(report.report_type)
166167
)
167168
return (
168169
self.env["py3o.report"]
169-
.create({"ir_actions_report_id": self.id})
170+
.create({"ir_actions_report_id": report.id})
170171
.create_report(res_ids, data)
171172
)
172173

report_py3o/models/py3o_report.py

Lines changed: 28 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,8 @@
1616
import pkg_resources
1717

1818
from odoo import _, api, fields, models, tools
19+
from odoo.exceptions import AccessError
20+
from odoo.tools.safe_eval import safe_eval, time
1921

2022
from ._py3o_parser_context import Py3oParserContext
2123

@@ -187,7 +189,9 @@ def _extend_parser_context(self, context, report_xml):
187189
def _get_parser_context(self, model_instance, data):
188190
report_xml = self.ir_actions_report_id
189191
context = Py3oParserContext(self.env).localcontext
190-
context.update(report_xml._get_rendering_context(model_instance.ids, data))
192+
context.update(
193+
report_xml._get_rendering_context(report_xml, model_instance.ids, data)
194+
)
191195
context["objects"] = model_instance
192196
self._extend_parser_context(context, report_xml)
193197
return context
@@ -199,9 +203,30 @@ def _postprocess_report(self, model_instance, result_path):
199203
# consumption...
200204
# ... but odoo wants the whole data in memory anyways :)
201205
buffer = BytesIO(f.read())
202-
self.ir_actions_report_id._postprocess_pdf_report(
203-
model_instance, buffer
206+
attachment_name = safe_eval(
207+
self.ir_actions_report_id.attachment,
208+
{"object": model_instance, "time": time},
204209
)
210+
if attachment_name:
211+
attachment_vals = {
212+
"name": attachment_name,
213+
"res_model": self.ir_actions_report_id.model,
214+
"res_id": model_instance.id,
215+
"raw": buffer.getvalue(),
216+
}
217+
try:
218+
attach = self.env["ir.attachment"].create(attachment_vals)
219+
except AccessError:
220+
logger.info(
221+
"Cannot save PDF report %s as attachment",
222+
attachment_vals["name"],
223+
)
224+
else:
225+
logger.info(
226+
"PDF document %s saved as attachment ID %d",
227+
attachment_vals["name"],
228+
attach.id,
229+
)
205230
return result_path
206231

207232
def _create_single_report(self, model_instance, data):

report_py3o/tests/test_report_py3o.py

Lines changed: 12 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -74,13 +74,13 @@ def _render_patched(self, result_text="test result", call_count=1):
7474
or result
7575
)
7676
# test the call the the create method inside our custom parser
77-
self.report._render(self.env.user.ids)
77+
self.report._render(self.report.id, self.env.user.ids)
7878
self.assertEqual(call_count, patched_pdf.call_count)
7979
# generated files no more exists
8080
self.assertFalse(os.path.exists(result))
8181

8282
def test_reports(self):
83-
res = self.report._render(self.env.user.ids)
83+
res = self.report._render(self.report.id, self.env.user.ids)
8484
self.assertTrue(res)
8585

8686
def test_reports_merge_zip(self):
@@ -93,7 +93,7 @@ def test_reports_merge_zip(self):
9393
py3o_report.__class__, "_zip_results"
9494
) as patched_zip_results:
9595
patched_zip_results.side_effect = _zip_results
96-
content, filetype = self.report._render(users.ids)
96+
content, filetype = self.report._render(self.report.id, users.ids)
9797
self.assertEqual(1, patched_zip_results.call_count)
9898
self.assertEqual(filetype, "zip")
9999

@@ -122,7 +122,7 @@ def test_report_load_from_attachment(self):
122122
# time we ask the report we received the saved attachment not a newly
123123
# generated document
124124
created_attachement.datas = base64.b64encode(b"new content")
125-
res = self.report._render(self.env.user.ids)
125+
res = self.report._render(self.report.id, self.env.user.ids)
126126
self.assertEqual((b"new content", self.report.py3o_filetype), res)
127127

128128
def test_report_post_process(self):
@@ -150,24 +150,24 @@ def test_report_template_configs(self):
150150
"odoo.addons.%s" % self.report.module, tmpl_name
151151
)
152152
self.assertTrue(os.path.exists(flbk_filename))
153-
res = self.report._render(self.env.user.ids)
153+
res = self.report._render(self.report.id, self.env.user.ids)
154154
self.assertTrue(res)
155155
# The generation fails if the template is not found
156156
self.report.module = False
157157
with self.assertRaises(TemplateNotFound), self.env.cr.savepoint():
158-
self.report._render(self.env.user.ids)
158+
self.report._render(self.report.id, self.env.user.ids)
159159

160160
# the template can also be provided as an abspath if it's root path
161161
# is trusted
162162
self.report.py3o_template_fallback = flbk_filename
163163
with self.assertRaises(TemplateNotFound):
164-
self.report._render(self.env.user.ids)
164+
self.report._render(self.report.id, self.env.user.ids)
165165
with temporary_copy(flbk_filename) as tmp_filename:
166166
self.report.py3o_template_fallback = tmp_filename
167167
tools.config.misc["report_py3o"] = {
168168
"root_tmpl_path": os.path.realpath(os.path.dirname(tmp_filename))
169169
}
170-
res = self.report._render(self.env.user.ids)
170+
res = self.report._render(self.report.id, self.env.user.ids)
171171
self.assertTrue(res)
172172

173173
# the tempalte can also be provided as a binary field
@@ -184,7 +184,7 @@ def test_report_template_configs(self):
184184
)
185185
self.report.py3o_template_id = py3o_template
186186
self.report.py3o_template_fallback = flbk_filename
187-
res = self.report._render(self.env.user.ids)
187+
res = self.report._render(self.report.id, self.env.user.ids)
188188
self.assertTrue(res)
189189

190190
@tools.misc.mute_logger("odoo.addons.report_py3o.models.py3o_report")
@@ -240,7 +240,7 @@ def test_py3o_report_availability(self):
240240
self.assertFalse(self.report.lo_bin_path)
241241
self.assertFalse(self.report.is_py3o_report_not_available)
242242
self.assertFalse(self.report.msg_py3o_report_not_available)
243-
res = self.report._render(self.env.user.ids)
243+
res = self.report._render(self.report.id, self.env.user.ids)
244244
self.assertTrue(res)
245245

246246
# The report should become unavailable for an non native output format
@@ -249,7 +249,7 @@ def test_py3o_report_availability(self):
249249
self.assertTrue(self.report.is_py3o_report_not_available)
250250
self.assertTrue(self.report.msg_py3o_report_not_available)
251251
with self.assertRaises(RuntimeError):
252-
self.report._render(self.env.user.ids)
252+
self.report._render(self.report.id, self.env.user.ids)
253253

254254
# if we reset the wrong path, everything should work
255255
self.env["ir.config_parameter"].set_param(
@@ -260,5 +260,5 @@ def test_py3o_report_availability(self):
260260
self.assertFalse(self.report.is_py3o_native_format)
261261
self.assertFalse(self.report.is_py3o_report_not_available)
262262
self.assertFalse(self.report.msg_py3o_report_not_available)
263-
res = self.report._render(self.env.user.ids)
263+
res = self.report._render(self.report.id, self.env.user.ids)
264264
self.assertTrue(res)

report_py3o/views/menu.xml

Lines changed: 0 additions & 4 deletions
This file was deleted.

report_py3o/views/py3o_template.xml

Lines changed: 4 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
<field name="name">py3o.template.configuration.search.view</field>
55
<field name="model">py3o.template</field>
66
<field name="arch" type="xml">
7-
<search string="Py3o Templates">
7+
<search>
88
<field name="name" />
99
<field name="filetype" />
1010
<group string="Group By" name="groupby">
@@ -21,7 +21,7 @@
2121
<field name="name">py3o.template.configuration.form.view</field>
2222
<field name="model">py3o.template</field>
2323
<field name="arch" type="xml">
24-
<form string="Py3o Templates">
24+
<form>
2525
<group name="main">
2626
<field name="name" />
2727
<field name="filetype" />
@@ -47,7 +47,8 @@
4747
</record>
4848
<menuitem
4949
id="py3o_template_configuration_menu"
50-
parent="py3o_config_menu"
50+
parent="base.reporting_menuitem"
5151
action="py3o_template_configuration_action"
52+
sequence="100"
5253
/>
5354
</odoo>

requirements.txt

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,7 @@
11
# generated from manifests external_dependencies
22
lxml
3+
py3o.formats
4+
py3o.template
5+
PyPDF2
36
xlrd
47
xlsxwriter
Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
../../../../report_py3o

0 commit comments

Comments
 (0)