# -*- coding: utf-8 -*- from odoo.tests import tagged from .common import TestPeEdiCommon, CODE_98_ERROR_MSG, MAX_WAIT_ITER import threading from unittest.mock import patch from datetime import timedelta from time import sleep @tagged('external_l10n', 'post_install', '-at_install', '-standard', 'external') class TestEdiIAP(TestPeEdiCommon): @classmethod def setUpClass(cls, chart_template_ref='pe', edi_format_ref='l10n_pe_edi.edi_pe_ubl_2_1'): super().setUpClass(chart_template_ref=chart_template_ref, edi_format_ref=edi_format_ref) cls.company_data['company'].l10n_pe_edi_provider = 'iap' def test_10_invoice_edi_flow(self): yesterday = self.certificate._get_pe_current_datetime().date() - timedelta(1) move = self._create_invoice(invoice_date=yesterday, date=yesterday) move.action_post() # Send doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) # Calls to IAP are disabled during testing, we need to remove the testing flag to let it perform the calls with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) self.assertRecordValues(doc, [{'error': False}]) self.assertRecordValues(move, [{'edi_state': 'sent'}]) # Cancel step 1 move.l10n_pe_edi_cancel_reason = 'abc' move.button_cancel_posted_moves() self.assertFalse(move.l10n_pe_edi_cancel_cdr_number) doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) self.assertTrue(move.l10n_pe_edi_cancel_cdr_number) self.assertRecordValues(move, [{'edi_state': 'to_cancel'}]) # Cancel step 2 doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) # We need to wait a bit before requesting the cancellation's status # to avoid getting a status code 98 (cancellation still being processed). for _ in range(MAX_WAIT_ITER): sleep(10) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) if not doc.error or doc.error != CODE_98_ERROR_MSG: break self.assertRecordValues(doc, [{'error': False}]) self.assertRecordValues(move, [{'edi_state': 'cancelled'}]) def test_20_refund_edi_flow(self): today = self.certificate._get_pe_current_datetime().date() move = self._create_refund(invoice_date=today, date=today) (move.reversed_entry_id + move).action_post() # Send doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) doc_reversed_entry = move.reversed_entry_id.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): (move.reversed_entry_id + move).action_process_edi_web_services(with_commit=False) self.assertRecordValues(doc, [{'error': False}]) self.assertTrue(doc_reversed_entry) self.assertRecordValues(doc_reversed_entry, [{'error': False}]) self.assertRecordValues(move, [{'edi_state': 'sent'}]) # Cancel step 1 move.l10n_pe_edi_cancel_reason = 'abc' move.button_cancel_posted_moves() self.assertFalse(move.l10n_pe_edi_cancel_cdr_number) doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) self.assertTrue(move.l10n_pe_edi_cancel_cdr_number) self.assertRecordValues(move, [{'edi_state': 'to_cancel'}]) # Cancel step 2 doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) # We need to wait a bit before requesting the cancellation's status # to avoid getting a status code 98 (cancellation still being processed). for _ in range(MAX_WAIT_ITER): sleep(10) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) if not doc.error or doc.error != CODE_98_ERROR_MSG: break self.assertRecordValues(doc, [{'error': False}]) self.assertRecordValues(move, [{'edi_state': 'cancelled'}]) def test_30_debit_note_edi_flow(self): today = self.certificate._get_pe_current_datetime().date() move = self._create_debit_note(invoice_date=today, date=today) (move.debit_origin_id + move).action_post() # Send doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) doc_debit_origin = move.debit_origin_id.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): (move.debit_origin_id + move).action_process_edi_web_services(with_commit=False) self.assertRecordValues(doc, [{'error': False}]) self.assertRecordValues(doc_debit_origin, [{'error': False}]) self.assertRecordValues(move, [{'edi_state': 'sent'}]) # Cancel step 1 move.l10n_pe_edi_cancel_reason = 'abc' move.button_cancel_posted_moves() self.assertFalse(move.l10n_pe_edi_cancel_cdr_number) doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) self.assertTrue(move.l10n_pe_edi_cancel_cdr_number) self.assertRecordValues(move, [{'edi_state': 'to_cancel'}]) # Cancel step 2 doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) # We need to wait a bit before requesting the cancellation's status # to avoid getting a status code 98 (cancellation still being processed). for _ in range(MAX_WAIT_ITER): sleep(10) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) if not doc.error or doc.error != CODE_98_ERROR_MSG: break self.assertRecordValues(doc, [{'error': False}]) self.assertRecordValues(move, [{'edi_state': 'cancelled'}]) def test_40_catch_error_in_cdr_cancel(self): """ Check that we correctly detect errors reported in the ResponseCode field of the CDR when cancelling an invoice. """ today = self.certificate._get_pe_current_datetime().date() yesterday = today - timedelta(1) move = self._create_invoice(invoice_date=yesterday, date=yesterday, name='F FFI-%s4' % self.time_name) move.action_post() doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) self.assertRecordValues(doc, [{'error': False}]) self.assertRecordValues(move, [{'edi_state': 'sent'}]) # Slightly tweak the cancellation request template so that SUNAT's response will contain an error in the CDR's ResponseCode. cancel_request_template = self.env.ref('l10n_pe_edi.ubl_pe_21_voided_documents') arch = cancel_request_template.arch arch_new = arch.replace('', '{}'.format(today)) cancel_request_template.write({'arch': arch_new}) # Cancel step 1 move.l10n_pe_edi_cancel_reason = 'abc' move.button_cancel_posted_moves() self.assertFalse(move.l10n_pe_edi_cancel_cdr_number) doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) self.assertTrue(move.l10n_pe_edi_cancel_cdr_number) self.assertRecordValues(move, [{'edi_state': 'to_cancel'}]) # Cancel step 2 doc = move.edi_document_ids.filtered(lambda d: d.state in ('to_send', 'to_cancel')) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) # We need to wait a bit before requesting the cancellation's status # to avoid getting a status code 98 (cancellation still being processed). for _ in range(MAX_WAIT_ITER): sleep(10) with patch.object(threading.current_thread(), 'testing', False): move.action_process_edi_web_services(with_commit=False) if doc.error and doc.error != CODE_98_ERROR_MSG: break expected_error = "

We got an error response from the OSE.

Original message:
2375|Fecha de emision del comprobante no coincide con la fecha de emision consignada en la comunicación" self.assertTrue(doc.error.startswith(expected_error), 'Error response: %s' % doc.error) self.assertRecordValues(move, [{'edi_state': 'to_cancel'}])