summaryrefslogtreecommitdiff
path: root/addons/snailmail/models/snailmail_letter.py
diff options
context:
space:
mode:
Diffstat (limited to 'addons/snailmail/models/snailmail_letter.py')
-rw-r--r--addons/snailmail/models/snailmail_letter.py406
1 files changed, 406 insertions, 0 deletions
diff --git a/addons/snailmail/models/snailmail_letter.py b/addons/snailmail/models/snailmail_letter.py
new file mode 100644
index 00000000..d4be829b
--- /dev/null
+++ b/addons/snailmail/models/snailmail_letter.py
@@ -0,0 +1,406 @@
+# -*- coding: utf-8 -*-
+# Part of Odoo. See LICENSE file for full copyright and licensing details.
+import re
+import base64
+
+from odoo import fields, models, api, _
+from odoo.addons.iap.tools import iap_tools
+from odoo.tools.safe_eval import safe_eval
+
+DEFAULT_ENDPOINT = 'https://iap-snailmail.odoo.com'
+PRINT_ENDPOINT = '/iap/snailmail/1/print'
+DEFAULT_TIMEOUT = 30
+
+ERROR_CODES = [
+ 'MISSING_REQUIRED_FIELDS',
+ 'CREDIT_ERROR',
+ 'TRIAL_ERROR',
+ 'NO_PRICE_AVAILABLE',
+ 'FORMAT_ERROR',
+ 'UNKNOWN_ERROR',
+]
+
+
+class SnailmailLetter(models.Model):
+ _name = 'snailmail.letter'
+ _description = 'Snailmail Letter'
+
+ user_id = fields.Many2one('res.users', 'Sent by')
+ model = fields.Char('Model', required=True)
+ res_id = fields.Integer('Document ID', required=True)
+ partner_id = fields.Many2one('res.partner', string='Recipient', required=True)
+ company_id = fields.Many2one('res.company', string='Company', required=True, readonly=True,
+ default=lambda self: self.env.company.id)
+ report_template = fields.Many2one('ir.actions.report', 'Optional report to print and attach')
+
+ attachment_id = fields.Many2one('ir.attachment', string='Attachment', ondelete='cascade')
+ attachment_datas = fields.Binary('Document', related='attachment_id.datas')
+ attachment_fname = fields.Char('Attachment Filename', related='attachment_id.name')
+ color = fields.Boolean(string='Color', default=lambda self: self.env.company.snailmail_color)
+ cover = fields.Boolean(string='Cover Page', default=lambda self: self.env.company.snailmail_cover)
+ duplex = fields.Boolean(string='Both side', default=lambda self: self.env.company.snailmail_duplex)
+ state = fields.Selection([
+ ('pending', 'In Queue'),
+ ('sent', 'Sent'),
+ ('error', 'Error'),
+ ('canceled', 'Canceled')
+ ], 'Status', readonly=True, copy=False, default='pending', required=True,
+ help="When a letter is created, the status is 'Pending'.\n"
+ "If the letter is correctly sent, the status goes in 'Sent',\n"
+ "If not, it will got in state 'Error' and the error message will be displayed in the field 'Error Message'.")
+ error_code = fields.Selection([(err_code, err_code) for err_code in ERROR_CODES], string="Error")
+ info_msg = fields.Char('Information')
+ display_name = fields.Char('Display Name', compute="_compute_display_name")
+
+ reference = fields.Char(string='Related Record', compute='_compute_reference', readonly=True, store=False)
+
+ message_id = fields.Many2one('mail.message', string="Snailmail Status Message")
+ notification_ids = fields.One2many('mail.notification', 'letter_id', "Notifications")
+
+ street = fields.Char('Street')
+ street2 = fields.Char('Street2')
+ zip = fields.Char('Zip')
+ city = fields.Char('City')
+ state_id = fields.Many2one("res.country.state", string='State')
+ country_id = fields.Many2one('res.country', string='Country')
+
+ @api.depends('reference', 'partner_id')
+ def _compute_display_name(self):
+ for letter in self:
+ if letter.attachment_id:
+ letter.display_name = "%s - %s" % (letter.attachment_id.name, letter.partner_id.name)
+ else:
+ letter.display_name = letter.partner_id.name
+
+ @api.depends('model', 'res_id')
+ def _compute_reference(self):
+ for res in self:
+ res.reference = "%s,%s" % (res.model, res.res_id)
+
+ @api.model
+ def create(self, vals):
+ msg_id = self.env[vals['model']].browse(vals['res_id']).message_post(
+ body=_("Letter sent by post with Snailmail"),
+ message_type='snailmail'
+ )
+
+ partner_id = self.env['res.partner'].browse(vals['partner_id'])
+ vals.update({
+ 'message_id': msg_id.id,
+ 'street': partner_id.street,
+ 'street2': partner_id.street2,
+ 'zip': partner_id.zip,
+ 'city': partner_id.city,
+ 'state_id': partner_id.state_id.id,
+ 'country_id': partner_id.country_id.id,
+ })
+ letter = super(SnailmailLetter, self).create(vals)
+
+ self.env['mail.notification'].sudo().create({
+ 'mail_message_id': msg_id.id,
+ 'res_partner_id': partner_id.id,
+ 'notification_type': 'snail',
+ 'letter_id': letter.id,
+ 'is_read': True, # discard Inbox notification
+ 'notification_status': 'ready',
+ })
+
+ return letter
+
+ def _fetch_attachment(self):
+ """
+ This method will check if we have any existent attachement matching the model
+ and res_ids and create them if not found.
+ """
+ self.ensure_one()
+ obj = self.env[self.model].browse(self.res_id)
+ if not self.attachment_id:
+ report = self.report_template
+ if not report:
+ report_name = self.env.context.get('report_name')
+ report = self.env['ir.actions.report']._get_report_from_name(report_name)
+ if not report:
+ return False
+ else:
+ self.write({'report_template': report.id})
+ # report = self.env.ref('account.account_invoices')
+ if report.print_report_name:
+ report_name = safe_eval(report.print_report_name, {'object': obj})
+ elif report.attachment:
+ report_name = safe_eval(report.attachment, {'object': obj})
+ else:
+ report_name = 'Document'
+ filename = "%s.%s" % (report_name, "pdf")
+ pdf_bin, _ = report.with_context(snailmail_layout=not self.cover)._render_qweb_pdf(self.res_id)
+ attachment = self.env['ir.attachment'].create({
+ 'name': filename,
+ 'datas': base64.b64encode(pdf_bin),
+ 'res_model': 'snailmail.letter',
+ 'res_id': self.id,
+ 'type': 'binary', # override default_type from context, possibly meant for another model!
+ })
+ self.write({'attachment_id': attachment.id})
+
+ return self.attachment_id
+
+ def _count_pages_pdf(self, bin_pdf):
+ """ Count the number of pages of the given pdf file.
+ :param bin_pdf : binary content of the pdf file
+ """
+ pages = 0
+ for match in re.compile(b"/Count\s+(\d+)").finditer(bin_pdf):
+ pages = int(match.group(1))
+ return pages
+
+ def _snailmail_create(self, route):
+ """
+ Create a dictionnary object to send to snailmail server.
+
+ :return: Dict in the form:
+ {
+ account_token: string, //IAP Account token of the user
+ documents: [{
+ pages: int,
+ pdf_bin: pdf file
+ res_id: int (client-side res_id),
+ res_model: char (client-side res_model),
+ address: {
+ name: char,
+ street: char,
+ street2: char (OPTIONAL),
+ zip: int,
+ city: char,
+ state: char (state code (OPTIONAL)),
+ country_code: char (country code)
+ }
+ return_address: {
+ name: char,
+ street: char,
+ street2: char (OPTIONAL),
+ zip: int,
+ city: char,at
+ state: char (state code (OPTIONAL)),
+ country_code: char (country code)
+ }
+ }],
+ options: {
+ color: boolean (true if color, false if black-white),
+ duplex: boolean (true if duplex, false otherwise),
+ currency_name: char
+ }
+ }
+ """
+ account_token = self.env['iap.account'].get('snailmail').account_token
+ dbuuid = self.env['ir.config_parameter'].sudo().get_param('database.uuid')
+ documents = []
+
+ batch = len(self) > 1
+ for letter in self:
+ document = {
+ # generic informations to send
+ 'letter_id': letter.id,
+ 'res_model': letter.model,
+ 'res_id': letter.res_id,
+ 'contact_address': letter.partner_id.with_context(snailmail_layout=True, show_address=True).name_get()[0][1],
+ 'address': {
+ 'name': letter.partner_id.name,
+ 'street': letter.partner_id.street,
+ 'street2': letter.partner_id.street2,
+ 'zip': letter.partner_id.zip,
+ 'state': letter.partner_id.state_id.code if letter.partner_id.state_id else False,
+ 'city': letter.partner_id.city,
+ 'country_code': letter.partner_id.country_id.code
+ },
+ 'return_address': {
+ 'name': letter.company_id.partner_id.name,
+ 'street': letter.company_id.partner_id.street,
+ 'street2': letter.company_id.partner_id.street2,
+ 'zip': letter.company_id.partner_id.zip,
+ 'state': letter.company_id.partner_id.state_id.code if letter.company_id.partner_id.state_id else False,
+ 'city': letter.company_id.partner_id.city,
+ 'country_code': letter.company_id.partner_id.country_id.code,
+ }
+ }
+ # Specific to each case:
+ # If we are estimating the price: 1 object = 1 page
+ # If we are printing -> attach the pdf
+ if route == 'estimate':
+ document.update(pages=1)
+ else:
+ # adding the web logo from the company for future possible customization
+ document.update({
+ 'company_logo': letter.company_id.logo_web and letter.company_id.logo_web.decode('utf-8') or False,
+ })
+ attachment = letter._fetch_attachment()
+ if attachment:
+ document.update({
+ 'pdf_bin': route == 'print' and attachment.datas.decode('utf-8'),
+ 'pages': route == 'estimate' and self._count_pages_pdf(base64.b64decode(attachment.datas)),
+ })
+ else:
+ letter.write({
+ 'info_msg': 'The attachment could not be generated.',
+ 'state': 'error',
+ 'error_code': 'ATTACHMENT_ERROR'
+ })
+ continue
+ if letter.company_id.external_report_layout_id == self.env.ref('l10n_de.external_layout_din5008', False):
+ document.update({
+ 'rightaddress': 0,
+ })
+ documents.append(document)
+
+ return {
+ 'account_token': account_token,
+ 'dbuuid': dbuuid,
+ 'documents': documents,
+ 'options': {
+ 'color': self and self[0].color,
+ 'cover': self and self[0].cover,
+ 'duplex': self and self[0].duplex,
+ 'currency_name': 'EUR',
+ },
+ # this will not raise the InsufficientCreditError which is the behaviour we want for now
+ 'batch': True,
+ }
+
+ def _get_error_message(self, error):
+ if error == 'CREDIT_ERROR':
+ link = self.env['iap.account'].get_credits_url(service_name='snailmail')
+ return _('You don\'t have enough credits to perform this operation.<br>Please go to your <a href=%s target="new">iap account</a>.', link)
+ if error == 'TRIAL_ERROR':
+ link = self.env['iap.account'].get_credits_url(service_name='snailmail', trial=True)
+ return _('You don\'t have an IAP account registered for this service.<br>Please go to <a href=%s target="new">iap.odoo.com</a> to claim your free credits.', link)
+ if error == 'NO_PRICE_AVAILABLE':
+ return _('The country of the partner is not covered by Snailmail.')
+ if error == 'MISSING_REQUIRED_FIELDS':
+ return _('One or more required fields are empty.')
+ if error == 'FORMAT_ERROR':
+ return _('The attachment of the letter could not be sent. Please check its content and contact the support if the problem persists.')
+ else:
+ return _('An unknown error happened. Please contact the support.')
+ return error
+
+ def _get_failure_type(self, error):
+ if error == 'CREDIT_ERROR':
+ return 'sn_credit'
+ if error == 'TRIAL_ERROR':
+ return 'sn_trial'
+ if error == 'NO_PRICE_AVAILABLE':
+ return 'sn_price'
+ if error == 'MISSING_REQUIRED_FIELDS':
+ return 'sn_fields'
+ if error == 'FORMAT_ERROR':
+ return 'sn_format'
+ else:
+ return 'sn_error'
+
+ def _snailmail_print(self, immediate=True):
+ valid_address_letters = self.filtered(lambda l: l._is_valid_address(l))
+ invalid_address_letters = self - valid_address_letters
+ invalid_address_letters._snailmail_print_invalid_address()
+ if valid_address_letters and immediate:
+ for letter in valid_address_letters:
+ letter._snailmail_print_valid_address()
+ self.env.cr.commit()
+
+ def _snailmail_print_invalid_address(self):
+ error = 'MISSING_REQUIRED_FIELDS'
+ error_message = _("The address of the recipient is not complete")
+ self.write({
+ 'state': 'error',
+ 'error_code': error,
+ 'info_msg': error_message,
+ })
+ self.notification_ids.sudo().write({
+ 'notification_status': 'exception',
+ 'failure_type': self._get_failure_type(error),
+ 'failure_reason': error_message,
+ })
+ self.message_id._notify_message_notification_update()
+
+ def _snailmail_print_valid_address(self):
+ """
+ get response
+ {
+ 'request_code': RESPONSE_OK, # because we receive 200 if good or fail
+ 'total_cost': total_cost,
+ 'credit_error': credit_error,
+ 'request': {
+ 'documents': documents,
+ 'options': options
+ }
+ }
+ }
+ """
+ endpoint = self.env['ir.config_parameter'].sudo().get_param('snailmail.endpoint', DEFAULT_ENDPOINT)
+ timeout = int(self.env['ir.config_parameter'].sudo().get_param('snailmail.timeout', DEFAULT_TIMEOUT))
+ params = self._snailmail_create('print')
+ response = iap_tools.iap_jsonrpc(endpoint + PRINT_ENDPOINT, params=params, timeout=timeout)
+ for doc in response['request']['documents']:
+ if doc.get('sent') and response['request_code'] == 200:
+ note = _('The document was correctly sent by post.<br>The tracking id is %s', doc['send_id'])
+ letter_data = {'info_msg': note, 'state': 'sent', 'error_code': False}
+ notification_data = {
+ 'notification_status': 'sent',
+ 'failure_type': False,
+ 'failure_reason': False,
+ }
+ else:
+ error = doc['error'] if response['request_code'] == 200 else response['reason']
+
+ note = _('An error occured when sending the document by post.<br>Error: %s', self._get_error_message(error))
+ letter_data = {
+ 'info_msg': note,
+ 'state': 'error',
+ 'error_code': error if error in ERROR_CODES else 'UNKNOWN_ERROR'
+ }
+ notification_data = {
+ 'notification_status': 'exception',
+ 'failure_type': self._get_failure_type(error),
+ 'failure_reason': note,
+ }
+
+ letter = self.browse(doc['letter_id'])
+ letter.write(letter_data)
+ letter.notification_ids.sudo().write(notification_data)
+ self.message_id._notify_message_notification_update()
+
+ def snailmail_print(self):
+ self.write({'state': 'pending'})
+ self.notification_ids.sudo().write({
+ 'notification_status': 'ready',
+ 'failure_type': False,
+ 'failure_reason': False,
+ })
+ self.message_id._notify_message_notification_update()
+ if len(self) == 1:
+ self._snailmail_print()
+
+ def cancel(self):
+ self.write({'state': 'canceled', 'error_code': False})
+ self.notification_ids.sudo().write({
+ 'notification_status': 'canceled',
+ })
+ self.message_id._notify_message_notification_update()
+
+ @api.model
+ def _snailmail_cron(self, autocommit=True):
+ letters_send = self.search([
+ '|',
+ ('state', '=', 'pending'),
+ '&',
+ ('state', '=', 'error'),
+ ('error_code', 'in', ['TRIAL_ERROR', 'CREDIT_ERROR', 'ATTACHMENT_ERROR', 'MISSING_REQUIRED_FIELDS'])
+ ])
+ for letter in letters_send:
+ letter._snailmail_print()
+ # Commit after every letter sent to avoid to send it again in case of a rollback
+ if autocommit:
+ self.env.cr.commit()
+
+ @api.model
+ def _is_valid_address(self, record):
+ record.ensure_one()
+ required_keys = ['street', 'city', 'zip', 'country_id']
+ return all(record[key] for key in required_keys)