OCA reporting engine fork for dev and update.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

95 lines
3.8 KiB

  1. # Copyright 2017 ACSONE SA/NV
  2. # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl)
  3. import json
  4. import mimetypes
  5. from werkzeug import exceptions
  6. from werkzeug.urls import url_decode
  7. from odoo.http import request, route
  8. from odoo.tools import html_escape
  9. from odoo.addons.web.controllers import main
  10. from odoo.addons.web.controllers.main import _serialize_exception, content_disposition
  11. class ReportController(main.ReportController):
  12. @route()
  13. def report_routes(self, reportname, docids=None, converter=None, **data):
  14. if converter != "py3o":
  15. return super(ReportController, self).report_routes(
  16. reportname=reportname, docids=docids, converter=converter, **data
  17. )
  18. context = dict(request.env.context)
  19. if docids:
  20. docids = [int(i) for i in docids.split(",")]
  21. if data.get("options"):
  22. data.update(json.loads(data.pop("options")))
  23. if data.get("context"):
  24. # Ignore 'lang' here, because the context in data is the
  25. # one from the webclient *but* if the user explicitely wants to
  26. # change the lang, this mechanism overwrites it.
  27. data["context"] = json.loads(data["context"])
  28. if data["context"].get("lang"):
  29. del data["context"]["lang"]
  30. context.update(data["context"])
  31. ir_action = request.env["ir.actions.report"]
  32. action_py3o_report = ir_action.get_from_report_name(
  33. reportname, "py3o"
  34. ).with_context(context)
  35. if not action_py3o_report:
  36. raise exceptions.HTTPException(
  37. description="Py3o action report not found for report_name "
  38. "%s" % reportname
  39. )
  40. res, filetype = action_py3o_report._render(docids, data)
  41. filename = action_py3o_report.gen_report_download_filename(docids, data)
  42. if not filename.endswith(filetype):
  43. filename = "{}.{}".format(filename, filetype)
  44. content_type = mimetypes.guess_type("x." + filetype)[0]
  45. http_headers = [
  46. ("Content-Type", content_type),
  47. ("Content-Length", len(res)),
  48. ("Content-Disposition", content_disposition(filename)),
  49. ]
  50. return request.make_response(res, headers=http_headers)
  51. @route()
  52. def report_download(self, data, token):
  53. """This function is used by 'qwebactionmanager.js' in order to trigger
  54. the download of a py3o/controller report.
  55. :param data: a javascript array JSON.stringified containg report
  56. internal url ([0]) and type [1]
  57. :returns: Response with a filetoken cookie and an attachment header
  58. """
  59. requestcontent = json.loads(data)
  60. url, report_type = requestcontent[0], requestcontent[1]
  61. if "py3o" not in report_type:
  62. return super(ReportController, self).report_download(data, token)
  63. try:
  64. reportname = url.split("/report/py3o/")[1].split("?")[0]
  65. docids = None
  66. if "/" in reportname:
  67. reportname, docids = reportname.split("/")
  68. if docids:
  69. # Generic report:
  70. response = self.report_routes(
  71. reportname, docids=docids, converter="py3o"
  72. )
  73. else:
  74. # Particular report:
  75. # decoding the args represented in JSON
  76. data = list(url_decode(url.split("?")[1]).items())
  77. response = self.report_routes(
  78. reportname, converter="py3o", **dict(data)
  79. )
  80. response.set_cookie("fileToken", token)
  81. return response
  82. except Exception as e:
  83. se = _serialize_exception(e)
  84. error = {"code": 200, "message": "Odoo Server Error", "data": se}
  85. return request.make_response(html_escape(json.dumps(error)))