From 66881591b02812378aafac4d3f91bdabe9e92b73 Mon Sep 17 00:00:00 2001 From: Santiago Rodriguez Date: Wed, 24 Jul 2024 12:02:55 -0600 Subject: [PATCH] [ADD] l10n_mx_cfdi_account --- l10n_mx_catalogs/__manifest__.py | 2 +- l10n_mx_cfdi/__manifest__.py | 2 +- l10n_mx_cfdi/security/ir.model.access.csv | 10 +- l10n_mx_cfdi/views/cfdi_documents_issued.xml | 2 +- l10n_mx_cfdi/views/cfdi_menu.xml | 7 +- l10n_mx_cfdi/views/product_template.xml | 19 +- l10n_mx_cfdi/views/res_partner.xml | 25 +- l10n_mx_cfdi_account/README.rst | 146 + l10n_mx_cfdi_account/__init__.py | 2 + l10n_mx_cfdi_account/__manifest__.py | 25 + l10n_mx_cfdi_account/i18n/l10n_mx_cfdi.pot | 2436 +++++++++++++++++ l10n_mx_cfdi_account/models/__init__.py | 11 + l10n_mx_cfdi_account/models/account_move.py | 700 +++++ .../models/account_move_line.py | 179 ++ .../models/account_move_reversal.py | 35 + .../models/account_partial_reconcile.py | 68 + .../models/account_payment.py | 207 ++ .../models/account_payment_register.py | 41 + l10n_mx_cfdi_account/models/account_tax.py | 18 + l10n_mx_cfdi_account/models/cfdi_document.py | 161 ++ .../models/res_config_settings.py | 12 + l10n_mx_cfdi_account/pyproject.toml | 3 + l10n_mx_cfdi_account/readme/CONFIGURE.md | 13 + l10n_mx_cfdi_account/readme/CONTRIBUTORS.md | 2 + l10n_mx_cfdi_account/readme/DESCRIPTION.md | 6 + l10n_mx_cfdi_account/readme/INSTALL.md | 2 + l10n_mx_cfdi_account/readme/ROADMAP.md | 2 + l10n_mx_cfdi_account/readme/USAGE.md | 20 + .../reports/report_external_layouts.xml | 423 +++ .../reports/report_invoice.xml | 139 + .../reports/report_payment.xml | 134 + .../security/ir.model.access.csv | 6 + .../static/description/icon.png | Bin 0 -> 3050 bytes .../static/description/icon.svg | 51 + .../static/description/index.html | 509 ++++ l10n_mx_cfdi_account/tests/__init__.py | 5 + .../tests/test_account_move.py | 10 + .../tests/test_account_move_line.py | 108 + .../tests/test_account_payment.py | 142 + .../tests/test_account_tax.py | 62 + l10n_mx_cfdi_account/views/account_move.xml | 139 + .../views/account_payment.xml | 42 + .../views/account_payment_register.xml | 12 + .../views/res_config_settings.xml | 22 + l10n_mx_cfdi_account/wizards/__init__.py | 3 + .../wizards/account_invoice_send_views.xml | 15 + .../wizards/create_cfdi_publico_en_general.py | 166 ++ .../create_cfdi_publico_en_general.xml | 64 + .../wizards/document_cancel.py | 77 + .../wizards/document_cancel_form.xml | 42 + .../wizards/download_cfdi_files.py | 79 + .../wizards/download_cfdi_files_wizard.xml | 42 + 52 files changed, 6416 insertions(+), 32 deletions(-) create mode 100644 l10n_mx_cfdi_account/README.rst create mode 100644 l10n_mx_cfdi_account/__init__.py create mode 100644 l10n_mx_cfdi_account/__manifest__.py create mode 100644 l10n_mx_cfdi_account/i18n/l10n_mx_cfdi.pot create mode 100644 l10n_mx_cfdi_account/models/__init__.py create mode 100644 l10n_mx_cfdi_account/models/account_move.py create mode 100644 l10n_mx_cfdi_account/models/account_move_line.py create mode 100644 l10n_mx_cfdi_account/models/account_move_reversal.py create mode 100644 l10n_mx_cfdi_account/models/account_partial_reconcile.py create mode 100644 l10n_mx_cfdi_account/models/account_payment.py create mode 100644 l10n_mx_cfdi_account/models/account_payment_register.py create mode 100644 l10n_mx_cfdi_account/models/account_tax.py create mode 100644 l10n_mx_cfdi_account/models/cfdi_document.py create mode 100644 l10n_mx_cfdi_account/models/res_config_settings.py create mode 100644 l10n_mx_cfdi_account/pyproject.toml create mode 100644 l10n_mx_cfdi_account/readme/CONFIGURE.md create mode 100644 l10n_mx_cfdi_account/readme/CONTRIBUTORS.md create mode 100644 l10n_mx_cfdi_account/readme/DESCRIPTION.md create mode 100644 l10n_mx_cfdi_account/readme/INSTALL.md create mode 100644 l10n_mx_cfdi_account/readme/ROADMAP.md create mode 100644 l10n_mx_cfdi_account/readme/USAGE.md create mode 100644 l10n_mx_cfdi_account/reports/report_external_layouts.xml create mode 100644 l10n_mx_cfdi_account/reports/report_invoice.xml create mode 100644 l10n_mx_cfdi_account/reports/report_payment.xml create mode 100644 l10n_mx_cfdi_account/security/ir.model.access.csv create mode 100644 l10n_mx_cfdi_account/static/description/icon.png create mode 100644 l10n_mx_cfdi_account/static/description/icon.svg create mode 100644 l10n_mx_cfdi_account/static/description/index.html create mode 100644 l10n_mx_cfdi_account/tests/__init__.py create mode 100644 l10n_mx_cfdi_account/tests/test_account_move.py create mode 100644 l10n_mx_cfdi_account/tests/test_account_move_line.py create mode 100644 l10n_mx_cfdi_account/tests/test_account_payment.py create mode 100644 l10n_mx_cfdi_account/tests/test_account_tax.py create mode 100644 l10n_mx_cfdi_account/views/account_move.xml create mode 100644 l10n_mx_cfdi_account/views/account_payment.xml create mode 100644 l10n_mx_cfdi_account/views/account_payment_register.xml create mode 100644 l10n_mx_cfdi_account/views/res_config_settings.xml create mode 100644 l10n_mx_cfdi_account/wizards/__init__.py create mode 100644 l10n_mx_cfdi_account/wizards/account_invoice_send_views.xml create mode 100644 l10n_mx_cfdi_account/wizards/create_cfdi_publico_en_general.py create mode 100644 l10n_mx_cfdi_account/wizards/create_cfdi_publico_en_general.xml create mode 100644 l10n_mx_cfdi_account/wizards/document_cancel.py create mode 100644 l10n_mx_cfdi_account/wizards/document_cancel_form.xml create mode 100644 l10n_mx_cfdi_account/wizards/download_cfdi_files.py create mode 100644 l10n_mx_cfdi_account/wizards/download_cfdi_files_wizard.xml diff --git a/l10n_mx_catalogs/__manifest__.py b/l10n_mx_catalogs/__manifest__.py index 2bf4369..b259b84 100644 --- a/l10n_mx_catalogs/__manifest__.py +++ b/l10n_mx_catalogs/__manifest__.py @@ -7,7 +7,7 @@ "category": "Localization", "version": "17.0.1.0.0", "license": "LGPL-3", - "depends": ["l10n_mx"], + "depends": ["base"], "data": [ "security/ir.model.access.csv", "data/l10n_mx_catalogs.c_clave_unidad.csv", diff --git a/l10n_mx_cfdi/__manifest__.py b/l10n_mx_cfdi/__manifest__.py index 482705c..4788a7b 100644 --- a/l10n_mx_cfdi/__manifest__.py +++ b/l10n_mx_cfdi/__manifest__.py @@ -7,7 +7,7 @@ "license": "LGPL-3", "category": "Accounting", "version": "17.0.1.0.0", - "depends": ["account", "l10n_mx_catalogs"], + "depends": ["l10n_mx_catalogs", "product"], "external_dependencies": { "python": ["facturama"], }, diff --git a/l10n_mx_cfdi/security/ir.model.access.csv b/l10n_mx_cfdi/security/ir.model.access.csv index a1690f0..af55fb4 100644 --- a/l10n_mx_cfdi/security/ir.model.access.csv +++ b/l10n_mx_cfdi/security/ir.model.access.csv @@ -1,11 +1,9 @@ id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink access_l10n_mx_cfdi_issuer,access_l10n_mx_cfdi_issuer,model_l10n_mx_cfdi_issuer,base.group_user,1,0,0,0 -manage_l10n_mx_cfdi_issuer,manage_l10n_mx_cfdi_issuer,model_l10n_mx_cfdi_issuer,account.group_account_manager,1,1,1,1 access_l10n_mx_cfdi_cfdi_service,access_l10n_mx_cfdi_cfdi_service,l10n_mx_cfdi.model_l10n_mx_cfdi_cfdi_service,base.group_user,1,0,0,0 manage_l10n_mx_cfdi_cfdi_service,manage_l10n_mx_cfdi_cfdi_service,l10n_mx_cfdi.model_l10n_mx_cfdi_cfdi_service,base.group_system,1,1,1,1 -l10n_mx_cfdi.access_l10n_mx_cfdi_cfdi_service_topup,access_l10n_mx_cfdi_cfdi_service_topup,l10n_mx_cfdi.model_l10n_mx_cfdi_cfdi_service_topup,base.group_user,1,0,0,0 -l10n_mx_cfdi.admin_l10n_mx_cfdi_cfdi_service_topup,admin_l10n_mx_cfdi_cfdi_service_topup,l10n_mx_cfdi.model_l10n_mx_cfdi_cfdi_service_topup,base.group_system,1,1,1,1 +access_l10n_mx_cfdi_cfdi_service_topup,access_l10n_mx_cfdi_cfdi_service_topup,model_l10n_mx_cfdi_cfdi_service_topup,base.group_user,1,0,0,0 +admin_l10n_mx_cfdi_cfdi_service_topup,admin_l10n_mx_cfdi_cfdi_service_topup,model_l10n_mx_cfdi_cfdi_service_topup,base.group_system,1,1,1,1 access_l10n_mx_cfdi_document,Acceso a certificados 4.0,model_l10n_mx_cfdi_document,base.group_user,1,1,1,1 -l10n_mx_cfdi.access_l10n_mx_cfdi_document_relation,access_l10n_mx_cfdi_document_relation,l10n_mx_cfdi.model_l10n_mx_cfdi_document_relation,base.group_user,1,1,1,1 -l10n_mx_cfdi.access_l10n_mx_cfdi_series,access_l10n_mx_cfdi_series,l10n_mx_cfdi.model_l10n_mx_cfdi_series,base.group_user,1,0,0,0 -l10n_mx_cfdi.manage_l10n_mx_cfdi_series,manage_l10n_mx_cfdi_series,l10n_mx_cfdi.model_l10n_mx_cfdi_series,account.group_account_manager,1,1,1,1 +access_l10n_mx_cfdi_document_relation,access_l10n_mx_cfdi_document_relation,model_l10n_mx_cfdi_document_relation,base.group_user,1,1,1,1 +access_l10n_mx_cfdi_series,access_l10n_mx_cfdi_series,model_l10n_mx_cfdi_series,base.group_user,1,0,0,0 diff --git a/l10n_mx_cfdi/views/cfdi_documents_issued.xml b/l10n_mx_cfdi/views/cfdi_documents_issued.xml index 86993dc..59ddba2 100644 --- a/l10n_mx_cfdi/views/cfdi_documents_issued.xml +++ b/l10n_mx_cfdi/views/cfdi_documents_issued.xml @@ -11,7 +11,7 @@ diff --git a/l10n_mx_cfdi/views/cfdi_menu.xml b/l10n_mx_cfdi/views/cfdi_menu.xml index 4125862..6720262 100644 --- a/l10n_mx_cfdi/views/cfdi_menu.xml +++ b/l10n_mx_cfdi/views/cfdi_menu.xml @@ -1,10 +1,5 @@ - + diff --git a/l10n_mx_cfdi/views/product_template.xml b/l10n_mx_cfdi/views/product_template.xml index 619847f..1d1e4cd 100644 --- a/l10n_mx_cfdi/views/product_template.xml +++ b/l10n_mx_cfdi/views/product_template.xml @@ -1,16 +1,19 @@ - Product Template: CFDI code and measurement unit fields + Product Template: CFDI code and measurement unit fields + product.template - - - - - + + + + + + + + + diff --git a/l10n_mx_cfdi/views/res_partner.xml b/l10n_mx_cfdi/views/res_partner.xml index fdf2c7c..b0ddd22 100644 --- a/l10n_mx_cfdi/views/res_partner.xml +++ b/l10n_mx_cfdi/views/res_partner.xml @@ -4,15 +4,22 @@ res.partner - - 1 - - - - - - - + + + + + + + + + + + + + + + + diff --git a/l10n_mx_cfdi_account/README.rst b/l10n_mx_cfdi_account/README.rst new file mode 100644 index 0000000..8779da9 --- /dev/null +++ b/l10n_mx_cfdi_account/README.rst @@ -0,0 +1,146 @@ +======================= +Mexico - CFDI - Account +======================= + +.. + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! source digest: sha256:26e47d930a46acf27c2c6e22f53f714adc3335be2cac745f521c692ad736dbcb + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png + :target: https://odoo-community.org/page/development-status + :alt: Beta +.. |badge2| image:: https://img.shields.io/badge/licence-LGPL--3-blue.png + :target: http://www.gnu.org/licenses/lgpl-3.0-standalone.html + :alt: License: LGPL-3 +.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Fl10n--mexico-lightgray.png?logo=github + :target: https://github.com/OCA/l10n-mexico/tree/17.0/l10n_mx_cfdi_account + :alt: OCA/l10n-mexico +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/l10n-mexico-17-0/l10n-mexico-17-0-l10n_mx_cfdi_account + :alt: Translate me on Weblate +.. |badge5| image:: https://img.shields.io/badge/runboat-Try%20me-875A7B.png + :target: https://runboat.odoo-community.org/builds?repo=OCA/l10n-mexico&target_branch=17.0 + :alt: Try me on Runboat + +|badge1| |badge2| |badge3| |badge4| |badge5| + +This module provides electronic invoicing for Mexico: + +- Generation of electronic invoices compliant with the CFDI 4.0 + standard. +- Customization of fiscal documents according to user needs. +- Centralized management of electronic invoices within Odoo. +- Tracking and recording of issued and received fiscal documents. + +**Table of contents** + +.. contents:: + :local: + +Installation +============ + +Prior to installing this module, you need to create and configure your +account with one of the `supported PAC listed +here `__. + +Configuration +============= + +1. In the invoicing app, go to Customers > customers + + 1. Create a new customer with its corresponding id number and fiscal + name + 2. In the contact form, go to Sales & purchase tab + 3. In the Fiscal Information section add the following information: + + - Fiscal regime + - Default CFDI usage + - Default payment method + - Default payment form + +2. In the invoicing app, go to Customers > Products + + 1. Open an existing product or create a new one and enter the general + information + 2. Go to the Accounting tab in the product form + 3. Add the product code and the unit of measure corresponding to + authority regulations + +Usage +===== + +Invoices +-------- + +- Go to Customers > Invoices and create a new record +- Fill in the previously created customer, the client’s previously + added fiscal information is essential for the stamping process. +- Add any product that has its product fiscal code +- Open the CFDI tab and add the payment method, and payment form +- Select the Confirm button, the invoice will be stamped with the + mexican authority +- Select the Send button to send the invoice to the customer + +Import CFDI +~~~~~~~~~~~ + +- Add the vendor bill xml file as an attachment in the chatter +- Click "Load from file" button + +Payments +-------- + +If the invoice payment method is PPD, the payment will be included in +the CFDI lines when registering a payment. + +Known issues / Roadmap +====================== + +- Cancel CFDI Documents +- Credit Notes + +Bug Tracker +=========== + +Bugs are tracked on `GitHub Issues `_. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us to smash it by providing a detailed and welcomed +`feedback `_. + +Do not contact contributors directly about support or help with technical issues. + +Credits +======= + +Authors +------- + +* Alexis López Zubieta (Auge TEC) + +Contributors +------------ + +- Alexis López Zubieta +- Maxime Chambreuil + +Maintainers +----------- + +This module is maintained by the OCA. + +.. image:: https://odoo-community.org/logo.png + :alt: Odoo Community Association + :target: https://odoo-community.org + +OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use. + +This module is part of the `OCA/l10n-mexico `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/l10n_mx_cfdi_account/__init__.py b/l10n_mx_cfdi_account/__init__.py new file mode 100644 index 0000000..aee8895 --- /dev/null +++ b/l10n_mx_cfdi_account/__init__.py @@ -0,0 +1,2 @@ +from . import models +from . import wizards diff --git a/l10n_mx_cfdi_account/__manifest__.py b/l10n_mx_cfdi_account/__manifest__.py new file mode 100644 index 0000000..8367d77 --- /dev/null +++ b/l10n_mx_cfdi_account/__manifest__.py @@ -0,0 +1,25 @@ +{ + "name": "Mexico - CFDI - Account", + "summary": "Mexico CFDI Account Integration", + "author": "Alexis López Zubieta (Auge TEC), " + "Odoo Community Association (OCA)", + "website": "https://github.com/OCA/l10n-mexico", + "license": "LGPL-3", + "category": "Accounting", + "version": "17.0.1.0.0", + "depends": ["l10n_mx_cfdi", "l10n_mx"], + "data": [ + "security/ir.model.access.csv", + "views/account_move.xml", + "views/account_payment_register.xml", + "views/account_payment.xml", + "views/res_config_settings.xml", + "wizards/document_cancel_form.xml", + "wizards/create_cfdi_publico_en_general.xml", + "wizards/account_invoice_send_views.xml", + "wizards/download_cfdi_files_wizard.xml", + "reports/report_external_layouts.xml", + "reports/report_invoice.xml", + "reports/report_payment.xml", + ], +} diff --git a/l10n_mx_cfdi_account/i18n/l10n_mx_cfdi.pot b/l10n_mx_cfdi_account/i18n/l10n_mx_cfdi.pot new file mode 100644 index 0000000..66622dd --- /dev/null +++ b/l10n_mx_cfdi_account/i18n/l10n_mx_cfdi.pot @@ -0,0 +1,2436 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * l10n_mx_cfdi +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 15.0\n" +"Report-Msgid-Bugs-To: \n" +"Last-Translator: \n" +"Language-Team: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: \n" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "" +"\n" +" Cadena Original del complemento de Certificación Digital del SAT\n" +" " +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "" +"\n" +" Sello Digital del CFDI\n" +" " +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "" +"\n" +" Sello Digital del SAT\n" +" " +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_receiver_data +msgid "Uso de CFDI:" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_payment_data +msgid "Moneda" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_payment_data +msgid "Método de Pago" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_id_data +msgid "Fecha / Hora de Emisión:" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "Fecha/Hora Certificación" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_id_data +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "Folio Fiscal:" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_related_documents +msgid "Folio Relacionado" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.external_layout_bold +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.external_layout_boxed +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.external_layout_standard +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.external_layout_striped +msgid "Folio:" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_payment_data +msgid "Forma de Pago" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_id_data +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "No. de Certificado Digital:" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "Número de Serie Certificado del SAT" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "RFC del PAC" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "Tipo de Comprobante:" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_related_documents +msgid "Tipo de Relación" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_related_documents +msgid "UUID" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_signature_data +msgid "Versión: 4.0" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_issuer_data +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_receiver_data +msgid "" +"
\n" +"\n" +" Régimen Fiscal:" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_receiver_data +msgid "" +"
\n" +"\n" +"
\n" +" Código Postal:" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.cfdi_issuer_data +msgid "" +"
\n" +"\n" +"
\n" +" Lugar de Expedición:" +msgstr "" + +#. module: l10n_mx_cfdi +#: model_terms:ir.ui.view,arch_db:l10n_mx_cfdi.view_cfdi_issuer +msgid "" +" 1: + # add 2h to now_utc_tz + now_utc_tz = now_utc_tz + timedelta(hours=2) + + # add time info to invoice_date + document_date = datetime.combine(document_date, now_utc_tz.time()) + + # invoice_date to ISO 8601 format + document_date_str = document_date.strftime("%Y-%m-%dT%H:%M:%S") + return document_date_str + + def gather_invoice_cfdi_items_data(self): + """ + Gather the data for the CFDI items + """ + self.ensure_one() + + cfdi_items_data = [] + for line in self.line_ids: + if not line.product_id: + continue + + cfdi_item_data = line._gater_cfdi_item_data() + cfdi_items_data.append(cfdi_item_data) + + return cfdi_items_data + + def gater_invoice_cfdi_item_data(self, line): + """Gather the data for a CFDI item. + :param line: The invoice line + :return: The CFDI item data + """ + + cfdi_item_data = line._gater_cfdi_item_data() + + return cfdi_item_data + + def validate_invoice_items_for_cfdi_generation(self): + err_msg = "" + # validate invoice items + for line in self.line_ids: + if not line.product_id: + continue + + if not line.product_id.l10n_mx_cfdi_product_code_id: + err_msg += ( + "- No se ha definido el código de producto para el producto %s\n" + % line.product_id.name + ) + + if not line.product_id.l10n_mx_cfdi_product_measurement_unit_id: + err_msg += ( + "- No se ha definido la unidad de medida para el producto %s\n" + % line.product_id.name + ) + + return err_msg + + @api.model + def _gather_invoice_cfdi_item_taxes_data(self, line, discount): + """Gather the taxes data for a CFDI item.""" + + price_unit_wo_discount = line.price_unit - discount + + taxes = [] + for tax_id in line.tax_ids: + computed_tax = tax_id.compute_all( + price_unit_wo_discount, + quantity=line.quantity, + currency=line.currency_id, + ) + tax_rate = ( + tax_id.amount / 100 + if tax_id.amount_type == "percent" + else tax_id.amount + ) + tax_total = ( + computed_tax["taxes"][0]["amount"] if computed_tax["taxes"] else 0 + ) + taxes.append( + { + "Name": tax_id.extract_l10n_mx_tax_code(), + "Rate": tax_rate, + "IsRetention": tax_id.extract_is_retention(), + "Base": computed_tax["total_excluded"], + "Total": tax_total, + } + ) + return taxes + + def prepare_invoice_cfdi_total_taxes(self): + self.ensure_one() + + total_taxes = {} + for line in self.line_ids: + if line.tax_line_id: + tax_id = line.tax_line_id + tax_code = tax_id.extract_l10n_mx_tax_code() + if not tax_code: + raise UserError( + _("The tax code for tax %s is not defined.") + % line.tax_ids[0].name + ) + + tax_rate = ( + tax_id.amount / 100 + if tax_id.amount_type == "percent" + else tax_id.amount + ) + + if tax_code in total_taxes: + total_taxes[tax_code]["Base"] += line.tax_base_amount + total_taxes[tax_code]["Total"] += line.price_total + else: + total_taxes[tax_code] = { + "Name": tax_code, + "Rate": tax_rate, + "IsRetention": tax_id.extract_is_retention(), + "Base": line.tax_base_amount, + "Total": line.price_total, + } + + # prepare float values to be serialized as JSON + for _k, v in total_taxes.items(): + v["Base"] = json_float_round(v["Base"], 2) + v["Total"] = json_float_round(v["Total"], 2) + + return list(total_taxes.values()) + + def button_draft(self): + for rec in self: + if rec.l10n_mx_cfdi_auto: + published_related_cfdi = rec.related_cert_ids.filtered_domain( + [("state", "=", "published")] + ) + if len(published_related_cfdi) > 0 and rec.move_type != "in_invoice": + # show CFDI cancel dialog + return ( + rec.env.ref("l10n_mx_cfdi_account.document_cancel_action") + .sudo() + .read()[0] + ) + + return super().button_draft() + + def create_refund_cfdi(self): + """ + Create CFDI of type 'E' (Egreso). + """ + for refund in self: + items_data = self.gather_invoice_cfdi_items_data() + + receivables = refund.line_ids.filtered( + lambda L: L.account_id.user_type_id.type == "receivable" + ) + partial_reconcile = self.env["account.partial.reconcile"].search( + [("debit_move_id", "in", receivables.ids)] + ) + partial_reconcile |= ( + receivables.matched_debit_ids + receivables.matched_credit_ids + ) + + move_lines = ( + partial_reconcile.credit_move_id + partial_reconcile.debit_move_id + ) + + related_cfdis = move_lines.move_id.related_cert_ids.filtered_domain( + [ + ("state", "=", "published"), + ("type", "=", "I"), + ] + ) + + cfdi_data = { + "NameId": "2", + "ExpeditionPlace": refund.issuer_id.zip, + "Date": self._format_cfdi_date_str(self.invoice_date), + "PaymentForm": refund.payment_form_id.code, + "PaymentMethod": refund.payment_method_id.code, + "Receiver": { + "Name": refund.partner_id.name, + "Rfc": refund.partner_id.vat, + "CfdiUse": refund.cfdi_use_id.code, + "FiscalRegime": refund.partner_id.tax_regime.code, + "TaxZipCode": refund.partner_id.zip, + }, + "Items": items_data, + "Relations": { + "Type": "01", + "Cfdis": [ + {"Uuid": related_cfdi.uuid} for related_cfdi in related_cfdis + ], + }, + } + + refund_cfdi = self.env["l10n_mx_cfdi.document"].create( + { + "type": "E", + "issuer_id": refund.issuer_id.id, + "receiver_id": refund.receiver_id.id, + "related_invoice_id": refund.id, + } + ) + + self._add_global_information_to_cfdi_if_required(cfdi_data) + + # register relations + refund_cfdi.update( + { + "related_document_ids": [ + ( + 0, + 0, + { + "source_id": refund_cfdi.id, + "target_id": related_cfdi.id, + "relation_type_id": self.env.ref( + "l10n_mx_catalogs.c_tipo_relacion_1" + ).id, + }, + ) + for related_cfdi in related_cfdis + ] + } + ) + + try: + refund_cfdi.publish(cfdi_data) + + refund.update( + { + "related_cert_ids": [(4, refund_cfdi.id)], + } + ) + + for cfdi in related_cfdis: + if cfdi.related_invoice_id: + cfdi.related_invoice_id.related_cert_ids |= refund_cfdi + + except Exception as e: + refund_cfdi.unlink() + raise e + + def _add_global_information_to_cfdi_if_required(self, cfdi_data): + if self.receiver_id.vat == "XAXX010101000": + currentDateTime = datetime.now() + + cfdi_data["GlobalInformation"] = { + "Periodicity": "01", # Daily periodicity + "Months": str(currentDateTime.month).rjust(2, "0"), + "Year": currentDateTime.year, + } + + cfdi_data["Receiver"]["TaxZipCode"] = self.issuer_id.zip + cfdi_data["Receiver"]["FiscalRegime"] = "616" + + @api.returns("self", lambda value: value.id) + def copy(self, default=None): + # avoid copying the related cfdis + default = (default or {}).update( + { + "related_cert_ids": [(6, 0, [])], + } + ) + + return super().copy(default) + + def _get_name_invoice_report(self): + self.ensure_one() + if self.company_id.account_fiscal_country_id.code == "MX": + return "l10n_mx_cfdi_account.report_invoice_document" + + return super()._get_name_invoice_report() + + def action_load_from_attachment(self): + self.ensure_one() + + # find xml attachment + xml_attachment = self.env["ir.attachment"].search( + [ + ("res_model", "=", "account.move"), + ("res_id", "=", self.id), + ("mimetype", "=", "application/xml"), + ], + limit=1, + ) + + if not xml_attachment: + raise UserError(_("No XML attachment found for this invoice.")) + + # decode attachment + xml = base64.b64decode(xml_attachment.datas) + + cfdi = self._parse_cfdi_xml(xml) + cfdi.xml_file = xml + + self.related_cert_ids |= cfdi + + def _parse_cfdi_xml(self, xml): + # parse CFDI XML + root = etree.fromstring(xml) + namespaces = root.nsmap + + # add tfd namespace + namespaces["tfd"] = "http://www.sat.gob.mx/TimbreFiscalDigital" + + cfdi_data = { + "type": root.attrib["TipoDeComprobante"], + "serie": root.attrib.get("Serie", ""), + "folio": root.attrib.get("Folio", ""), + "state": "published", + "related_invoice_id": self.id, + } + + # get uuid + timbre_fiscal = root.find( + "./cfdi:Complemento/tfd:TimbreFiscalDigital", namespaces + ) + cfdi_data["uuid"] = timbre_fiscal.attrib["UUID"] + + issuer_id = self._resolve_issuer_from_xml(namespaces, root) + cfdi_data["issuer_id"] = issuer_id.id + self.issuer_id = issuer_id + + receiver_id, cfdi_use = self._resolve_receiver_data_from_xml(namespaces, root) + cfdi_data["receiver_id"] = receiver_id.id + cfdi_use_model = self.env["l10n_mx_catalogs.c_uso_cfdi"] + cfdi_use = cfdi_use_model.search([("code", "=", cfdi_use)], limit=1) + + self.receiver_id = receiver_id + self.cfdi_use_id = cfdi_use + + # create or update cfdi document + cfdi_document_model = self.env["l10n_mx_cfdi.document"] + document = cfdi_document_model.search( + [("uuid", "=", cfdi_data["uuid"])], limit=1 + ) + if document: + document.write(cfdi_data) + else: + document = cfdi_document_model.create(cfdi_data) + + self.cfdi_document_id = document + self.cfdi_required = True + + # resolve payment form + payment_form_model = self.env["l10n_mx_catalogs.c_forma_pago"] + payment_form_code = root.attrib["FormaPago"] + self.payment_form_id = payment_form_model.search( + [("code", "=", payment_form_code)], limit=1 + ) + + # resolve payment method + payment_method_model = self.env["l10n_mx_catalogs.c_metodo_pago"] + payment_method_code = root.attrib["MetodoPago"] + self.payment_method_id = payment_method_model.search( + [("code", "=", payment_method_code)], limit=1 + ) + + return document + + def _resolve_receiver_data_from_xml(self, namespaces, root): + # get receiver + receiver = root.find("cfdi:Receptor", namespaces) + receiver_id = self.env["res.partner"].search( + [("vat", "=", receiver.attrib["Rfc"])], limit=1 + ) + if not receiver_id: + raise UserError( + _("Cannot find the receptor of the certificate. RFC: %s") + % receiver.attrib["Rfc"] + ) + + cfdi_use = receiver.attrib["UsoCFDI"] + return receiver_id, cfdi_use + + def _resolve_issuer_from_xml(self, namespaces, root): + # get issuer + issuer = root.find("cfdi:Emisor", namespaces) + issuer_id = self.env["l10n_mx_cfdi.issuer"].search( + [("vat", "=", issuer.attrib["Rfc"])] + ) + if not issuer_id: + # find partner + partner_id = self.env["res.partner"].search( + [("vat", "=", issuer.attrib["Rfc"])] + ) + if not partner_id: + raise UserError( + _("Cannot find the partner who emitted the certificate. " "RFC: %s") + % issuer.attrib["Rfc"] + ) + + # create issuer + issuer_id = self.env["l10n_mx_cfdi.issuer"].create( + { + "partner_id": partner_id.id, + } + ) + return issuer_id + + def action_generate_cfdi(self): + self.ensure_one() + + if self.cfdi_document_id.state == "published": + raise UserError(_("The CFDI has been published.")) + + if self.move_type == "out_invoice": + self.create_invoice_cfdi() + + if self.move_type == "out_refund": + # create credit note CFDI if required + if self.amount_residual != 0: + raise UserError( + _( + "You cannot generate a CFDI for a credit note with a " + "pending amount." + ) + ) + self.create_refund_cfdi() diff --git a/l10n_mx_cfdi_account/models/account_move_line.py b/l10n_mx_cfdi_account/models/account_move_line.py new file mode 100644 index 0000000..ff77424 --- /dev/null +++ b/l10n_mx_cfdi_account/models/account_move_line.py @@ -0,0 +1,179 @@ +from odoo import api, fields, models +from odoo.tools import float_is_zero, float_round, json_float_round + + +class AccountMoveLine(models.Model): + _inherit = "account.move.line" + + cfdi_price_unit = fields.Monetary( + compute="_compute_cfdi_fields", + store=True, + ) + + cfdi_subtotal = fields.Monetary( + compute="_compute_cfdi_fields", + store=True, + ) + + cfdi_discount = fields.Monetary( + compute="_compute_cfdi_fields", + store=True, + ) + + @api.depends( + "product_id", + "price_unit", + "quantity", + "discount", + ) + def _compute_cfdi_fields(self): + for line in self: + line._gater_cfdi_item_data() + + def _gater_cfdi_item_data(self): + self.ensure_one() + + res = {} + currency = self.currency_id.sudo() + + # use product price decimal precision to round the price calculations + price_decimal_precision = self.env.ref("product.decimal_price").sudo().digits + + # Compute 'Subtotal'. + line_discount_price_unit = self.price_unit + + if hasattr(self, "discount_fixed"): + line_discount_price_unit -= self.discount_fixed + + line_discount_price_unit = line_discount_price_unit * ( + 1 - self.discount / 100.0 + ) + # round the price unit to the currency precision to prevent + # differences between the invoice totals and the CFDI total + line_discount_price_unit = float_round( + line_discount_price_unit, precision_digits=price_decimal_precision + ) + + subtotal = self.quantity * line_discount_price_unit + + # keep track of taxes included in price to subtract them later + # from the unit price as the CFDI specification doesn't support + # then + taxes_included = 0 + + cfdi_taxes = [] + if self.tax_ids: + # Compute taxes and adjust 'Subtotal' and 'Total' + taxes = self.tax_ids._origin.with_context(force_sign=1) + taxes_res = taxes.compute_all( + line_discount_price_unit, + quantity=self.quantity, + currency=self.currency_id, + product=self.product_id, + partner=self.partner_id, + is_refund=self.move_id.move_type in ("out_refund", "in_refund"), + ) + res["Subtotal"] = taxes_res["total_excluded"] + res["Total"] = taxes_res["total_included"] + + for computed_tax in taxes_res["taxes"]: + tax_id = self.env["account.tax"].browse(computed_tax["id"]) + tax_rate = ( + tax_id.amount / 100 + if tax_id.amount_type == "percent" + else tax_id.amount + ) + is_retention = tax_id.extract_is_retention() + tax_rate = json_float_round(tax_rate, precision_digits=6) + tax_total = json_float_round( + computed_tax["amount"], precision_digits=currency.decimal_places + ) + tax_base = json_float_round( + taxes_res["total_excluded"], + precision_digits=currency.decimal_places, + ) + + # SAT requires positive retention taxes, but Odoo uses negative values. + if is_retention: + tax_rate *= -1 + tax_total *= -1 + + cfdi_taxes.append( + { + "Name": tax_id.extract_l10n_mx_tax_code(), + "Rate": tax_rate, + "IsRetention": is_retention, + "Base": tax_base, + "Total": tax_total, + } + ) + + if tax_id.price_include: + taxes_included += tax_total + + if cfdi_taxes: + res.update( + { + "Taxes": cfdi_taxes, + "TaxObject": "02", # 'Si objeto de impuesto' + } + ) + else: + res["Total"] = res["Subtotal"] = subtotal + res["TaxObject"] = "01" + + if self.product_id.default_code: + res["IdentificationNumber"] = self.product_id.default_code + unit_included_taxes = taxes_included / (self.quantity or 1) + line_discount_price_unit -= unit_included_taxes + res.update( + { + "Quantity": self.quantity, + "ProductCode": self.product_id.l10n_mx_cfdi_product_code_id.code, + "Description": self.name, + "UnitCode": ( + self.product_id.l10n_mx_cfdi_product_measurement_unit_id.code + ), + } + ) + + self._round_values_to_currency_precision(res) + + # compute discount + expected_subtotal_wo_discount = line_discount_price_unit * self.quantity + discount = ( + (self.price_unit * self.quantity) + - expected_subtotal_wo_discount + - taxes_included + ) + if float_is_zero(discount, precision_digits=currency.decimal_places): + # ignore a difference below the currency precision + discount = 0 + + res["Discount"] = discount + res["Subtotal"] += discount + + # recompute the unit price from the subtotal to avoid rounding + res["UnitPrice"] = res["Subtotal"] / (self.quantity or 1) + + self._round_values_to_currency_precision(res) + + # store the values to be used in the report + self.cfdi_subtotal = res["Subtotal"] + self.cfdi_discount = res["Discount"] + self.cfdi_price_unit = res["UnitPrice"] + + return res + + def _round_values_to_currency_precision(self, res, skip=None): + currency_decimal_places = self.currency_id.decimal_places + + # Round all values to the currency precision + for k, v in res.items(): + if skip and k in skip: + continue + + if isinstance(v, float): + res[k] = json_float_round(v, precision_digits=currency_decimal_places) + else: + res[k] = v diff --git a/l10n_mx_cfdi_account/models/account_move_reversal.py b/l10n_mx_cfdi_account/models/account_move_reversal.py new file mode 100644 index 0000000..5205cea --- /dev/null +++ b/l10n_mx_cfdi_account/models/account_move_reversal.py @@ -0,0 +1,35 @@ +from odoo import models + + +class AccountMoveReversal(models.TransientModel): + _inherit = "account.move.reversal" + + def _prepare_default_reversal(self, move): + """Add CFDI required fields to the reversal if the original move is a CFDI""" + + res = super()._prepare_default_reversal(move) + + if move.cfdi_required: + data = { + "cfdi_required": True, + "payment_method_id": self.env["l10n_mx_catalogs.c_metodo_pago"] + .search([("code", "=", "PUE")]) + .id, + "cfdi_use_id": self.env["l10n_mx_catalogs.c_uso_cfdi"] + .search([("code", "=", "G02")]) + .id, + "issuer_id": move.issuer_id.id, + "related_cert_ids": [(6, 0, [])], + } + + # set the right cfdi use for operations with public + if move.partner_id.vat == "XAXX010101000": + data["cfdi_use_id"] = ( + self.env["l10n_mx_catalogs.c_uso_cfdi"] + .search([("code", "=", "S01")]) + .id + ) + + res.update(data) + + return res diff --git a/l10n_mx_cfdi_account/models/account_partial_reconcile.py b/l10n_mx_cfdi_account/models/account_partial_reconcile.py new file mode 100644 index 0000000..8f8246d --- /dev/null +++ b/l10n_mx_cfdi_account/models/account_partial_reconcile.py @@ -0,0 +1,68 @@ +from odoo import models + + +class AccountPartialReconcile(models.Model): + _inherit = "account.partial.reconcile" + + def create(self, vals_list): + """Create Payments and Credit Note CFDI if required""" + + res = super().create(vals_list) + + if self.env.company.l10n_mx_cfdi_auto: + move_line_ids = res.debit_move_id | res.credit_move_id + + for move in move_line_ids.move_id: + if move.move_type == "entry": + # create payment CFDI if required + payment = move.payment_id + payment_requires_cfdi = any( + invoice.cfdi_required + and invoice.payment_method_id.code == "PPD" + for invoice in payment.reconciled_invoice_ids + ) + + if ( + payment.payment_type == "inbound" + and payment.is_reconciled + and payment_requires_cfdi + ): + payment.create_payment_cfdi() + + if move.move_type == "out_refund": + # create credit note CFDI if required + existent_cfdi = move.related_cert_ids.filtered_domain( + [("type", "=", "E"), ("state", "=", "published")] + ) + + if ( + move.amount_residual == 0 + and move.cfdi_required + and not existent_cfdi + ): + move.create_refund_cfdi() + + return res + + def unlink(self): + """Cancel related Payments CFDI if any""" + + move_line_ids = self.debit_move_id | self.credit_move_id + res = super().unlink() + + if self.env.company.l10n_mx_cfdi_auto: + for move in move_line_ids.move_id: + if move.move_type == "entry": + payment = move.payment_id + related_cfdi = payment.related_cert_ids.filtered_domain( + [("type", "=", "P"), ("state", "=", "published")] + ) + if related_cfdi and not payment.is_reconciled: + payment.cancel_payment_cfdi() + + if move.move_type == "out_refund": + for cfdi in move.related_cert_ids: + if cfdi.state == "published" and cfdi.type == "E": + cfdi.cancel("02") + + return res diff --git a/l10n_mx_cfdi_account/models/account_payment.py b/l10n_mx_cfdi_account/models/account_payment.py new file mode 100644 index 0000000..7daf71b --- /dev/null +++ b/l10n_mx_cfdi_account/models/account_payment.py @@ -0,0 +1,207 @@ +from datetime import datetime + +from odoo import _, models +from odoo.exceptions import ValidationError +from odoo.tools import json_float_round + + +class AccountPayment(models.Model): + _inherit = "account.payment" + + def action_generate_cfdi(self): + self.ensure_one() + + if self.cfdi_document_id: + raise ValidationError(_("The payment already has a related CFDI.")) + + if not self.is_reconciled: + raise ValidationError(_("The payment is not fully reconciled.")) + + if self.payment_type == "inbound": + self.create_payment_cfdi() + + def create_payment_cfdi(self): + """ + Create payment-type ('P') CFDI matching invoice payments if needed. + """ + + self.ensure_one() + + # assert move type is inbound payment + if self.move_type != "entry" or self.payment_type != "inbound": + raise ValidationError(_("You can only create customer payments.")) + + # check if the payment is fully reconciled + if not self.is_reconciled: + raise ValidationError(_("The payment is not fully reconciled.")) + + payment_data = self.prepare_payment_cfdi() + + # get invoices issuer + issuer = self.reconciled_invoice_ids.issuer_id + issuer.ensure_one() + + # get invoices receiver + receiver = self.reconciled_invoice_ids.receiver_id + if not receiver: + # resolve receiver from the invoice CFDI for legacy invoices + receiver = self.reconciled_invoice_ids.related_cert_ids.filtered_domain( + [("type", "=", "I"), ("state", "=", "published")] + ).mapped("receiver_id")[0] + + receiver.ensure_one() + + payment_cfdi = self.env["l10n_mx_cfdi.document"].create( + { + "type": "P", + "issuer_id": issuer.id, + "receiver_id": receiver.id, + "related_payment_id": self.id, + } + ) + + # establecer uso de CFDI a Pagos (CP01) + self.cfdi_use_id = self.env.ref("l10n_mx_catalogs.c_uso_cfdi_CP01").id + + try: + cfdi_data = { + "ExpeditionPlace": issuer.zip, + "Receiver": { + "Name": receiver.name, + "Rfc": receiver.vat, + "CfdiUse": self.cfdi_use_id.code, + "FiscalRegime": receiver.tax_regime.code, + "TaxZipCode": receiver.zip, + }, + "Complemento": {"Payments": [payment_data]}, + } + + if receiver.vat == "XAXX010101000": + currentDateTime = datetime.now() + + cfdi_data["GlobalInformation"] = { + "Periodicity": "01", # Daily periodicity + "Months": str(currentDateTime.month).rjust(2, "0"), + "Year": currentDateTime.year, + } + + cfdi_data["Receiver"]["TaxZipCode"] = issuer.zip + cfdi_data["Receiver"]["FiscalRegime"] = "616" + + payment_cfdi.publish(cfdi_data) + + self.update( + { + "related_cert_ids": [(4, payment_cfdi.id)], + "cfdi_document_id": payment_cfdi.id, + } + ) + + for invoice in self.reconciled_invoice_ids: + invoice.related_cert_ids |= payment_cfdi + + except Exception as e: + payment_cfdi.unlink() + raise e + + def prepare_payment_cfdi(self): + self.ensure_one() + + related_documents_data = [] + + for invoice in self.reconciled_invoice_ids: + if not invoice.cfdi_document_id: + raise ValidationError( + _( + "Error al emitir CFDI tipo Comprobante de Pago. " + "La factura %s no tiene CFDI" + ) + % invoice.name + ) + + # get related cfdi of type 'Ingreso' + existent_invoice_cfdi = invoice.related_cert_ids.filtered_domain( + [("type", "=", "I"), ("state", "=", "published")] + ) + existent_invoice_cfdi.ensure_one() + + # get related CFDIs of type 'Pago' + existent_payments_cfdi = invoice.related_cert_ids.filtered_domain( + [("type", "=", "P"), ("state", "=", "published")] + ) + + # initialize related document data + related_document_data = { + "Uuid": existent_invoice_cfdi.uuid, + "Folio": existent_invoice_cfdi.name, + "PartialityNumber": len(existent_payments_cfdi) + 1, + "PaymentMethod": "PUE", + "AmountPaid": 0, + "PreviousBalanceAmount": invoice.amount_residual, + } + + # add amounts from matched credit lines + for credit in invoice.line_ids.matched_credit_ids: + # add line amount if it comes from the current payment + if credit.credit_move_id.move_id == self.move_id: + related_document_data["AmountPaid"] += credit.amount + related_document_data["PreviousBalanceAmount"] += credit.amount + + # add tax data + tax_data = self._compute_taxes(related_document_data["AmountPaid"], invoice) + if tax_data: + related_document_data["TaxObject"] = "02" + related_document_data["Taxes"] = tax_data + else: + related_document_data["TaxObject"] = "01" + + # round monetary fields + for field in ["AmountPaid", "PreviousBalanceAmount"]: + related_document_data[field] = json_float_round( + related_document_data[field], 2 + ) + + # add related document data to the list + related_documents_data.append(related_document_data) + + payment_date = self.move_id._format_cfdi_date_str(self.date) + payment_data = { + "Date": payment_date, + "PaymentForm": self.move_id.payment_form_id.code, + "Amount": json_float_round(self.amount, 2), + "RelatedDocuments": related_documents_data, + } + return payment_data + + def _compute_taxes(self, amount_paid, invoice): + total_taxes = invoice.prepare_invoice_cfdi_total_taxes() + payment_taxes = [] + + # skip if there are no taxes + if not total_taxes: + return payment_taxes + + # compute taxes base (amount_paid = rate * base) so (base = amount_paid / rate) + total_rate = sum(float(tax["Rate"] + 1) for tax in total_taxes) + base = amount_paid / total_rate + for tax in total_taxes: + payment_taxes.append( + { + "Name": tax["Name"], + "Rate": tax["Rate"], + "IsRetention": tax["IsRetention"], + "Base": json_float_round(base, 2), + "Total": json_float_round(base * float(tax["Rate"]), 2), + } + ) + + return payment_taxes + + def cancel_payment_cfdi(self): + self.ensure_one() + + for cfdi in self.related_cert_ids: + if cfdi.state == "published": + cfdi.cancel( + "02" + ) # cancel reason: 'Comprobantes emitidos con errores sin relación' diff --git a/l10n_mx_cfdi_account/models/account_payment_register.py b/l10n_mx_cfdi_account/models/account_payment_register.py new file mode 100644 index 0000000..b6d0c1c --- /dev/null +++ b/l10n_mx_cfdi_account/models/account_payment_register.py @@ -0,0 +1,41 @@ +from odoo import _, fields, models +from odoo.exceptions import UserError + + +class AccountPaymentRegister(models.TransientModel): + _inherit = "account.payment.register" + + payment_form_id = fields.Many2one( + "l10n_mx_catalogs.c_forma_pago", string="Forma de Pago", required=True + ) + + def _init_payments(self, to_process, edit_mode=False): + """ + Add payment for id to payments creation data + """ + for entry in to_process: + entry["create_vals"].update( + { + "payment_form_id": self.payment_form_id.id, + } + ) + + return super()._init_payments(to_process, edit_mode) + + def _create_payments(self): + # Prevent partial payments on invoices with cfdi and payment + # method different of 'PPD' + if self.payment_difference > 0: + related_invoices = self.line_ids.move_id + if any( + invoice.cfdi_required and invoice.payment_method_id.code != "PPD" + for invoice in related_invoices + ): + raise UserError( + _( + "You cannot register a partial payment against an " + "invoice with a CFDI and PUE as the payment method." + ) + ) + + return super()._create_payments() diff --git a/l10n_mx_cfdi_account/models/account_tax.py b/l10n_mx_cfdi_account/models/account_tax.py new file mode 100644 index 0000000..5b033ed --- /dev/null +++ b/l10n_mx_cfdi_account/models/account_tax.py @@ -0,0 +1,18 @@ +from odoo import models + + +class AccountTax(models.Model): + _inherit = "account.tax" + + def extract_l10n_mx_tax_code(self): + self.ensure_one() + if "ISR" in self.name: + return "ISR" + elif "IEPS" in self.name: + return "IEPS" + else: + return "IVA" + + def extract_is_retention(self): + self.ensure_one() + return "RET" in self.name diff --git a/l10n_mx_cfdi_account/models/cfdi_document.py b/l10n_mx_cfdi_account/models/cfdi_document.py new file mode 100644 index 0000000..2f6427c --- /dev/null +++ b/l10n_mx_cfdi_account/models/cfdi_document.py @@ -0,0 +1,161 @@ +import base64 +import re + +from odoo import _, api, fields, models +from odoo.exceptions import UserError + + +class Document(models.Model): + _inherit = "l10n_mx_cfdi.document" + _description = "CFDI document" + + ### + # Certificate fields + ### + + related_invoice_id = fields.Many2one( + "account.move", string="Factura relacionada", readonly=True + ) + related_payment_id = fields.Many2one( + "account.payment", string="Pago relacionado", readonly=True + ) + + ### + # Computed fields generation functions + ### + + @api.depends("tracking_id") + def _compute_download_files_if_needed(self): + for entry in self: + if entry.tracking_id: + if not entry.pdf_file: + report_type, report, resource_ids = self._resolve_report() + + if report: + # force the report to be rendered to work around a bug + # in _render_qweb_pdf + report = self.env["ir.actions.report"].with_context( + force_report_rendering=True + ) + doc_data, doc_format = report._render_qweb_pdf( + report_type, resource_ids + ) + # in some scenarios, the report is not generated, + # so we need to check if the file is empty + if doc_data: + result = base64.b64encode(doc_data) + entry.pdf_file = result + + if not entry.pdf_file: + # fallback to the provider's PDF + res = entry.issuer_id.service_id.sudo().get_cfdi_pdf( + entry.tracking_id + ) + entry.pdf_file = res["Content"] + + # set filename + entry.pdf_filename = "%s.pdf" % entry.name + + if not entry.xml_file: + res = entry.issuer_id.service_id.sudo().get_cfdi_xml( + entry.tracking_id + ) + entry.xml_file = res["Content"].encode("utf-8") + entry.xml_filename = "%s.xml" % entry.name + + entry.files_in_cache = True + else: + entry.files_in_cache = False + + def _resolve_report(self): + """Returns the report and resource IDs for generating the PDF file.""" + report_type = None + report = None + resource_ids = [] + + for document in self: + if document.type in ("I", "E") and document.related_invoice_id: + report_type = "account.account_invoices" + report = self.env.ref(report_type) + resource_ids = [document.related_invoice_id.id] + + if document.type == "P" and document.related_payment_id: + report_type = "account.action_report_payment_receipt" + report = self.env.ref(report_type) + resource_ids = [document.related_payment_id.id] + + return report_type, report, resource_ids + + ### + # Model methods + ### + + def create(self, vals_list): + # Set values to serie and folio from sequence if not provided + + # check if vals_list is a list of dictionaries + if isinstance(vals_list, dict): + vals_list = [vals_list] + + for vals in vals_list: + if "serie" not in vals or "folio" not in vals: + issuer = self._resolve_issuer_on_create(vals) + if ( + issuer.use_origin_document_sequence + and vals.get("type", False) != "T" + and vals.get("is_global_note", False) is False + ): + self._set_serie_and_folio_from_document_sequence(vals) + else: + self._set_serie_and_folio_from_cfdi_sequence(vals) + + # Create certificate + return super().create(vals_list) + + def _set_serie_and_folio_from_document_sequence(self, vals): + serie = "" + folio = "" + document_name = "" + + if "related_invoice_id" in vals: + invoice = self.env["account.move"].browse(vals["related_invoice_id"]) + document_name = invoice.name + + if "related_payment_id" in vals: + payment = self.env["account.payment"].browse(vals["related_payment_id"]) + document_name = payment.name + + if not document_name: + raise UserError(_("Unable to determine the origin document name.")) + + # extract numeric postfix from invoice name using regex + match = re.search(r"\d+$", document_name) + if match: + folio = match.group() + serie = document_name[: -len(match.group())] + else: + raise UserError(_("Invoice name does not contain a numeric postfix.")) + + # remove non-alphanumeric characters from serie + serie = re.sub(r"\W+", "", serie) + + vals["serie"] = serie + vals["folio"] = folio + + def action_check_status(self): + self.ensure_one() + + service = self.issuer_id.service_id.sudo() + amount_total = 0 + if self.related_invoice_id: + amount_total = self.related_invoice_id.amount_total + + if self.related_payment_id: + amount_total = self.related_payment_id.amount + + status = service.check_cfdi_status( + self.uuid, self.issuer_id.vat, self.receiver_id.vat, amount_total + ) + + if self.state != status: + self.state = status diff --git a/l10n_mx_cfdi_account/models/res_config_settings.py b/l10n_mx_cfdi_account/models/res_config_settings.py new file mode 100644 index 0000000..cd408ff --- /dev/null +++ b/l10n_mx_cfdi_account/models/res_config_settings.py @@ -0,0 +1,12 @@ +from odoo import fields, models + + +class ResConfigSettings(models.TransientModel): + _inherit = "res.config.settings" + + l10n_mx_cfdi_auto = fields.Boolean( + "Create CFDI on post", + related="company_id.l10n_mx_cfdi_auto", + help="Enable to automatically sign CFDI when validating invoices.", + readonly=False, + ) diff --git a/l10n_mx_cfdi_account/pyproject.toml b/l10n_mx_cfdi_account/pyproject.toml new file mode 100644 index 0000000..4231d0c --- /dev/null +++ b/l10n_mx_cfdi_account/pyproject.toml @@ -0,0 +1,3 @@ +[build-system] +requires = ["whool"] +build-backend = "whool.buildapi" diff --git a/l10n_mx_cfdi_account/readme/CONFIGURE.md b/l10n_mx_cfdi_account/readme/CONFIGURE.md new file mode 100644 index 0000000..0203851 --- /dev/null +++ b/l10n_mx_cfdi_account/readme/CONFIGURE.md @@ -0,0 +1,13 @@ +1. In the invoicing app, go to Customers > customers + 1. Create a new customer with its corresponding id number and fiscal name + 2. In the contact form, go to Sales & purchase tab + 3. In the Fiscal Information section add the following information: + - Fiscal regime + - Default CFDI usage + - Default payment method + - Default payment form +2. In the invoicing app, go to Customers > Products + 1. Open an existing product or create a new one and enter the general + information + 2. Go to the Accounting tab in the product form + 3. Add the product code and the unit of measure corresponding to authority regulations diff --git a/l10n_mx_cfdi_account/readme/CONTRIBUTORS.md b/l10n_mx_cfdi_account/readme/CONTRIBUTORS.md new file mode 100644 index 0000000..44cc131 --- /dev/null +++ b/l10n_mx_cfdi_account/readme/CONTRIBUTORS.md @@ -0,0 +1,2 @@ +- Alexis López Zubieta \<\> +- Maxime Chambreuil \<\> diff --git a/l10n_mx_cfdi_account/readme/DESCRIPTION.md b/l10n_mx_cfdi_account/readme/DESCRIPTION.md new file mode 100644 index 0000000..b38103f --- /dev/null +++ b/l10n_mx_cfdi_account/readme/DESCRIPTION.md @@ -0,0 +1,6 @@ +This module provides electronic invoicing for Mexico: + +- Generation of electronic invoices compliant with the CFDI 4.0 standard. +- Customization of fiscal documents according to user needs. +- Centralized management of electronic invoices within Odoo. +- Tracking and recording of issued and received fiscal documents. diff --git a/l10n_mx_cfdi_account/readme/INSTALL.md b/l10n_mx_cfdi_account/readme/INSTALL.md new file mode 100644 index 0000000..a775043 --- /dev/null +++ b/l10n_mx_cfdi_account/readme/INSTALL.md @@ -0,0 +1,2 @@ +Prior to installing this module, you need to create and configure your +account with one of the [supported PAC listed here](https://github.com/OCA/l10n-mexico/blob/17.0/l10n_mx_cfdi/README.md). diff --git a/l10n_mx_cfdi_account/readme/ROADMAP.md b/l10n_mx_cfdi_account/readme/ROADMAP.md new file mode 100644 index 0000000..4be7dda --- /dev/null +++ b/l10n_mx_cfdi_account/readme/ROADMAP.md @@ -0,0 +1,2 @@ +- Cancel CFDI Documents +- Credit Notes diff --git a/l10n_mx_cfdi_account/readme/USAGE.md b/l10n_mx_cfdi_account/readme/USAGE.md new file mode 100644 index 0000000..50cefc7 --- /dev/null +++ b/l10n_mx_cfdi_account/readme/USAGE.md @@ -0,0 +1,20 @@ +## Invoices + +- Go to Customers > Invoices and create a new record +- Fill in the previously created customer, the client’s previously added + fiscal information is essential for the stamping process. +- Add any product that has its product fiscal code +- Open the CFDI tab and add the payment method, and payment form +- Select the Confirm button, the invoice will be stamped with the mexican + authority +- Select the Send button to send the invoice to the customer + +### Import CFDI + +- Add the vendor bill xml file as an attachment in the chatter +- Click "Load from file" button + +## Payments + +If the invoice payment method is PPD, the payment will be included in the +CFDI lines when registering a payment. diff --git a/l10n_mx_cfdi_account/reports/report_external_layouts.xml b/l10n_mx_cfdi_account/reports/report_external_layouts.xml new file mode 100644 index 0000000..206f0d8 --- /dev/null +++ b/l10n_mx_cfdi_account/reports/report_external_layouts.xml @@ -0,0 +1,423 @@ + + + + + + + + + + + diff --git a/l10n_mx_cfdi_account/reports/report_invoice.xml b/l10n_mx_cfdi_account/reports/report_invoice.xml new file mode 100644 index 0000000..96d6765 --- /dev/null +++ b/l10n_mx_cfdi_account/reports/report_invoice.xml @@ -0,0 +1,139 @@ + + + + + + + + + + diff --git a/l10n_mx_cfdi_account/reports/report_payment.xml b/l10n_mx_cfdi_account/reports/report_payment.xml new file mode 100644 index 0000000..3199229 --- /dev/null +++ b/l10n_mx_cfdi_account/reports/report_payment.xml @@ -0,0 +1,134 @@ + + + + + + + diff --git a/l10n_mx_cfdi_account/security/ir.model.access.csv b/l10n_mx_cfdi_account/security/ir.model.access.csv new file mode 100644 index 0000000..00c0e6c --- /dev/null +++ b/l10n_mx_cfdi_account/security/ir.model.access.csv @@ -0,0 +1,6 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +cancel_l10n_mx_cfdi_document,Cancelar a certificados,model_l10n_mx_cfdi_account_document_cancel,base.group_user,1,1,1,1 +access_l10n_mx_cfdi_generic_invoice_create,access_l10n_mx_cfdi_generic_invoice_create,model_l10n_mx_cfdi_account_generic_invoice_create,base.group_user,1,1,1,1 +access_l10n_mx_cfdi_download_cfdi_files_wizard,access_l10n_mx_cfdi_download_cfdi_files_wizard,model_l10n_mx_cfdi_account_dl_cfdi_files_wizard,base.group_user,1,1,1,1 +manage_l10n_mx_cfdi_issuer,manage_l10n_mx_cfdi_issuer,l10n_mx_cfdi.model_l10n_mx_cfdi_issuer,account.group_account_manager,1,1,1,1 +manage_l10n_mx_cfdi_series,manage_l10n_mx_cfdi_series,l10n_mx_cfdi.model_l10n_mx_cfdi_series,account.group_account_manager,1,1,1,1 diff --git a/l10n_mx_cfdi_account/static/description/icon.png b/l10n_mx_cfdi_account/static/description/icon.png new file mode 100644 index 0000000000000000000000000000000000000000..70392fd3e60441f5b7e2a84f72b92b790080401c GIT binary patch literal 3050 zcmV!Oo;QS zTX&k&O{VGeF_}*P=uF*aI@6i_(ayB}ukB2lw39!YOs9>L%$RsGagvGSynxuoNgGHE z=3zkCNKos6gic6sI^A6dlo3Lbu^xMHhxrVDaHqY!y{~qEyT9ME(!_!ZeYsDco(TI#}J1>%&C4-V(!+ zEE5k_mPhdn<&*;%igl^T>&jZb{r9x}?7Kh7^+9I;_!r(an`Po7W8;5)^ub(ju`y(5 zyit^wRS|atf-)#YX0~l#L1d7R=jB!V zS3T1c8}uBT1pLuF&vT6XhG*!5k@GL#7{95v??3j;pG(W_3K1kXAj`)vZr{zTH%|Y_ zKRY%thpNz6&&9sc@yDP2*M}OtDer9S8`h}Lah!8{hT~!|id+zo<)bJ%I)A0Eou25a z?jHK8UJnpn^xqhN@A+p>{O%ulXIo!yOFLs97#O_yKTp9$E(n;mj|+_mQNOyZ%@aEs z2rSIaEsnZ*dwEHDg>LM#cbyZ%5bs40_<;k>2M#t9#AZelV9JNnb$wuYiq1kp6co!ee&_!kyZ_TlKTE@e>nA~{jjy^{s%ug^UB_8ow|s9`2HWFD4Iw9 z9o_gig%YC1voUgC9-FIJWQIWd9C(06N zT}Ny4<0S8%c70{2X{u>`_^sMtTa`&q?hct-I`6TM&kyat`-pAV;rueEi|FJ-6h+nQ zA_PJD`Yv~Mone?z@~5RnLwm=)8`GaalaHh{dTYnKA3XmZZJ@o_SIb=bZTR&zm7{)F ztBtt%{>LvEEUf_KS^H8z5k#xi>GcKxfXnGR^~$Tm!z0N_;{8c;OOw?73Cw{@nYr%i zr(W>f9MN4p{YBF=XSFAbY)2!#Ck^_Fpa1j?y{;VMy;g?kDx=X%k|fKr@1Hw=_Ut<; z{mIUqR;l|Fm|xCvo6?l;Y5u`3yLV>B9h$zSkZ*Fch8VwNdDATw1>qAnMx$KtVukvzhBobz$tkdN}`jA714<>&q z2m-{CC_}!KX|G0OX=i6gep&H6%qXEU8n53UfRz>8+NT)XmohMx3CfTU$MNn@dR(p< z8I)xW8Ejuk6q}toqoe)GTH5^f&2BT*jbT0kcLYJ`R)#=>m%4Ica^}M)x-KQ3*|T?bv>alXsj2AmO7mKig$Qn;G6k3gcFboq2WU~4=h9KF758~-4N@Xw@ zBawxrrKKz{+yFAjM-UXprA~O}`B;8|;=)P9X~<=A9`urAhL!}24DzYf8iJsb&j=6oCZcs>@3$!HLh2cgiy!oq5AX0`6b)f*z8C<-hK zIY74YU>KI%d!Q)PnE~s3#X!0x zToi6_A`fVqe)Jnh34(x|EkWQGi`il^4-5=BrweifGuLE6;M0eG8&AtU`=D!=Ok-oA zQ#&%tCx>Rc?5sf&)vA0v zALG)(dGdrLiJXRk;e|*zEXigKU&|-wW}8mi{r8#8xy{pm{k1MBO7iupPf zE+7hW`KECt#s&KO`D?qFf*_FGWd|Ctpeky0f!``oTwfzFPUcKvDZB`IbYDX;M|3^Osl zW}3Xm?Fk0g>@@VnfBTaUS(f$r)=YJVh4N}2hG9KD7tWnK|Ep(CA_&seb;jfNXtny} z(aH3*^Y4FqF>we0C@Ika0Kcv%()MY!I;~bWJUn7DnPQxf9>`Lwi{h)cq(+hQ#p&(2 z005wAjTF@?OB5`hR;v>PVH34e{JM_K)lS*9ZyjoCst+D&O?;j{vVnyn<*TSDuWzXJ zxV_sseWb;a!qEr;6ivOR{Ig8uA#NmqVlO$d*t$cZ` z932`Q9vr+XS&1OCz4>NAz?Uy&6MBd^LFSLZRC+!moV*0Ie3z z+iQwaO-@WsPE49iCJL(C;Bk8az%UFer0U7t>nJWR;W#c9<8oa=zCQReBmkhaw9rlJ zWGx>6z-TlrE=E{(3D%@*05FPDk)%qaDK6Z?D8T21_}Xk`-$WYXs=zzT>mW%L+!d8~ zkbF2!B(IW(!kr=?02oQC;AaaxM;U^mC`GCA+w*vR?g;q^f|6)Qimfjnj^kNwp^^t% zTs{B*NlLf{_7<0qqSR8Dyg17^32tuN(ICkujmX>{}D2k!b!gJ5P(9`qj s_V4507LA5(X=!e6KXUBYqX>fhANw<28gaN%r~m)}07*qoM6N<$g2c`Ic>n+a literal 0 HcmV?d00001 diff --git a/l10n_mx_cfdi_account/static/description/icon.svg b/l10n_mx_cfdi_account/static/description/icon.svg new file mode 100644 index 0000000..c868bbe --- /dev/null +++ b/l10n_mx_cfdi_account/static/description/icon.svg @@ -0,0 +1,51 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/l10n_mx_cfdi_account/static/description/index.html b/l10n_mx_cfdi_account/static/description/index.html new file mode 100644 index 0000000..12141a4 --- /dev/null +++ b/l10n_mx_cfdi_account/static/description/index.html @@ -0,0 +1,509 @@ + + + + + +Mexico - CFDI - Account + + + +
+

Mexico - CFDI - Account

+ + +

Beta License: LGPL-3 OCA/l10n-mexico Translate me on Weblate Try me on Runboat

+

This module provides electronic invoicing for Mexico:

+
    +
  • Generation of electronic invoices compliant with the CFDI 4.0 +standard.
  • +
  • Customization of fiscal documents according to user needs.
  • +
  • Centralized management of electronic invoices within Odoo.
  • +
  • Tracking and recording of issued and received fiscal documents.
  • +
+

Table of contents

+ +
+

Installation

+

Prior to installing this module, you need to create and configure your +account with one of the supported PAC listed +here.

+
+
+

Configuration

+
    +
  1. In the invoicing app, go to Customers > customers
      +
    1. Create a new customer with its corresponding id number and fiscal +name
    2. +
    3. In the contact form, go to Sales & purchase tab
    4. +
    5. In the Fiscal Information section add the following information:
        +
      • Fiscal regime
      • +
      • Default CFDI usage
      • +
      • Default payment method
      • +
      • Default payment form
      • +
      +
    6. +
    +
  2. +
  3. In the invoicing app, go to Customers > Products
      +
    1. Open an existing product or create a new one and enter the general +information
    2. +
    3. Go to the Accounting tab in the product form
    4. +
    5. Add the product code and the unit of measure corresponding to +authority regulations
    6. +
    +
  4. +
+
+
+

Usage

+
+

Invoices

+
    +
  • Go to Customers > Invoices and create a new record
  • +
  • Fill in the previously created customer, the client’s previously +added fiscal information is essential for the stamping process.
  • +
  • Add any product that has its product fiscal code
  • +
  • Open the CFDI tab and add the payment method, and payment form
  • +
  • Select the Confirm button, the invoice will be stamped with the +mexican authority
  • +
  • Select the Send button to send the invoice to the customer
  • +
+
+

Import CFDI

+
    +
  • Add the vendor bill xml file as an attachment in the chatter
  • +
  • Click “Load from file” button
  • +
+
+
+
+

Payments

+

If the invoice payment method is PPD, the payment will be included in +the CFDI lines when registering a payment.

+
+
+
+

Known issues / Roadmap

+
    +
  • Cancel CFDI Documents
  • +
  • Credit Notes
  • +
+
+
+

Bug Tracker

+

Bugs are tracked on GitHub Issues. +In case of trouble, please check there if your issue has already been reported. +If you spotted it first, help us to smash it by providing a detailed and welcomed +feedback.

+

Do not contact contributors directly about support or help with technical issues.

+
+
+

Credits

+
+

Authors

+ +
+
+

Contributors

+ +
+
+

Maintainers

+

This module is maintained by the OCA.

+ +Odoo Community Association + +

OCA, or the Odoo Community Association, is a nonprofit organization whose +mission is to support the collaborative development of Odoo features and +promote its widespread use.

+

This module is part of the OCA/l10n-mexico project on GitHub.

+

You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.

+
+
+
+ + diff --git a/l10n_mx_cfdi_account/tests/__init__.py b/l10n_mx_cfdi_account/tests/__init__.py new file mode 100644 index 0000000..92d08e2 --- /dev/null +++ b/l10n_mx_cfdi_account/tests/__init__.py @@ -0,0 +1,5 @@ +# from . import test_account_move +from . import test_account_move_line + +# from . import test_account_payment +from . import test_account_tax diff --git a/l10n_mx_cfdi_account/tests/test_account_move.py b/l10n_mx_cfdi_account/tests/test_account_move.py new file mode 100644 index 0000000..cbbd3ff --- /dev/null +++ b/l10n_mx_cfdi_account/tests/test_account_move.py @@ -0,0 +1,10 @@ +# Copyright (C) 2019 Brian McMaster +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from odoo.addons.account.tests.common import AccountTestInvoicingCommon + + +class TestAccountMove(AccountTestInvoicingCommon): + @classmethod + def setUpClass(cls): + super().setUpClass() diff --git a/l10n_mx_cfdi_account/tests/test_account_move_line.py b/l10n_mx_cfdi_account/tests/test_account_move_line.py new file mode 100644 index 0000000..71134e6 --- /dev/null +++ b/l10n_mx_cfdi_account/tests/test_account_move_line.py @@ -0,0 +1,108 @@ +from odoo.tests import TransactionCase + + +class TestAccountMoveLine(TransactionCase): + def setUp(self): + super().setUp() + + self.partner = self.env["res.partner"].create( + { + "name": "Test Partner", + } + ) + + self.account = self.env["account.account"].search( + [("account_type", "=", "income")], + limit=1, + ) + + self.account_iva = self.env["account.account"].create( + { + "name": "IVA account", + "code": "10", + "account_type": "liability_current", + } + ) + + self.tax_group = self.env["account.tax.group"].create( + { + "name": "IVA", + "country_id": self.env.ref("base.us").id, + } + ) + + self.tax = self.env["account.tax"].create( + { + "name": "IVA", + "amount": 16.00, + "amount_type": "percent", + "type_tax_use": "sale", + "country_id": self.env.ref("base.us").id, + "tax_group_id": self.tax_group.id, + "invoice_repartition_line_ids": [ + (0, 0, {"repartition_type": "base"}), + ( + 0, + 0, + { + "factor_percent": 100.0, + "repartition_type": "tax", + "account_id": self.account_iva.id, + }, + ), + ], + } + ) + + self.product = self.env["product.template"].create( + { + "name": "Test Product", + "default_code": "TP", + "list_price": "100", + } + ) + + self.journal = self.env["account.journal"].search( + [("type", "=", "sale")], + limit=1, + ) + + self.move_line_1_vals = { + "product_id": self.env["product.product"] + .search( + [("default_code", "=", "TP")], + limit=1, + ) + .id, + "name": "Test Move Line 1", + "quantity": 2, + "account_id": self.account.id, + "price_unit": 100.00, + "display_type": "product", + "tax_ids": [(6, 0, [self.tax.id])], + } + + self.move_vals = { + "name": "Test Move", + "move_type": "out_invoice", + "journal_id": self.journal.id, + "partner_id": self.partner.id, + "invoice_line_ids": [(0, 0, self.move_line_1_vals)], + } + + def test_compute_cfdi_fields(self): + move = self.env["account.move"].create(self.move_vals) + + move_line_1 = move.invoice_line_ids.search( + [("name", "=", "Test Move Line 1")], + limit=1, + ) + + move_line_1._compute_cfdi_fields() + + # Assuming the test logic for computing CFDI fields here... + + self.assertEqual(move_line_1.cfdi_subtotal, 200.00) + self.assertEqual(move_line_1.cfdi_price_unit, 100.00) + + # Add assertions for move_line_2 if necessary diff --git a/l10n_mx_cfdi_account/tests/test_account_payment.py b/l10n_mx_cfdi_account/tests/test_account_payment.py new file mode 100644 index 0000000..6a57805 --- /dev/null +++ b/l10n_mx_cfdi_account/tests/test_account_payment.py @@ -0,0 +1,142 @@ +# from odoo.exceptions import ValidationError +# from odoo.tests import TransactionCase +# +# +# class TestAccountPayment(TransactionCase): +# +# def setUp(self): +# super().setUp() +# +# # Create test data +# self.partner = self.env["res.partner"].create( +# { +# "name": "Test Partner", +# "vat": "TESTVAT", +# "zip": "12345", # Add other required fields +# } +# ) +# +# self.payment_journal = self.env["account.journal"].create( +# { +# "name": "Test Journal", +# "code": "TEST", +# "type": "sale", +# } +# ) +# +# self.payment_method = self.env["account.payment.method"].create( +# { +# "name": "Test Payment Method", +# "code": "TEST_PM", +# "payment_type": "inbound", # Adjust payment type as needed +# } +# ) +# +# self.payment_method_line = self.env["account.payment.method.line"].create( +# { +# "payment_method_id": self.payment_method.id, +# "payment_type": "inbound", # Adjust payment type as needed +# "journal_id": self.payment_journal.id, +# "sequence": 1, +# # Add other required fields +# } +# ) +# +# self.payment = self.env["account.payment"].create( +# { +# "partner_id": self.partner.id, +# "journal_id": self.payment_journal.id, +# "amount": 100.0, +# "payment_type": "inbound", +# "payment_method_line_id": self.payment_method_line.id, +# } +# ) +# +# self.issuer = self.env["res.partner"].create( +# { +# "name": "Issuer", +# "vat": "ISSUERVAT", +# "zip": "54321", # Add other required fields +# } +# ) +# +# self.invoice = self.env["account.move"].create( +# { +# "partner_id": self.partner.id, +# "type": "out_invoice", # Example type, adjust as needed +# "issuer_id": self.issuer.id, +# "amount_total": 50.0, # Example amount, adjust as needed +# # Add other required fields +# } +# ) +# +# self.payment.reconciled_invoice_ids = [(4, self.invoice.id)] +# +# def test_action_generate_cfdi_with_existing_cfdi(self): +# # Add a related CFDI to the payment +# self.payment.write( +# {"cfdi_document_id": self.env["l10n_mx_cfdi.document"].create({}).id} +# ) +# +# # Try to generate CFDI again, it should raise validation error +# with self.assertRaises(ValidationError): +# self.payment.action_generate_cfdi() +# +# def test_action_generate_cfdi_not_fully_reconciled(self): +# # Try to generate CFDI for a payment that is not fully reconciled +# with self.assertRaises(ValidationError): +# self.payment.action_generate_cfdi() +# +# def test_create_payment_cfdi(self): +# # Create a fully reconciled payment +# self.payment.move_type = "entry" +# self.payment.is_reconciled = True +# self.payment.create_payment_cfdi() +# +# # Check if the payment has a related CFDI +# self.assertTrue(self.payment.cfdi_document_id) +# self.assertEqual(self.payment.cfdi_document_id.type, "P") +# self.assertEqual(self.payment.cfdi_document_id.issuer_id.zip, "12345") +# self.assertEqual(self.payment.cfdi_document_id.receiver_id.vat, "TESTVAT") +# +# def test_create_payment_cfdi_with_legacy_invoice(self): +# # Create a fully reconciled payment with a legacy invoice +# self.payment.move_type = "entry" +# self.payment.is_reconciled = True +# self.payment.reconciled_invoice_ids = [ +# ( +# 0, +# 0, +# { +# "related_cert_ids": [ +# ( +# 0, +# 0, +# { +# "type": "I", +# "state": "published", +# "receiver_id": self.partner.id, +# }, +# ) +# ] +# }, +# ) +# ] +# +# # Now, receiver should be resolved from invoice CFDI +# self.payment.create_payment_cfdi() +# +# # Check if the payment has a related CFDI +# self.assertTrue(self.payment.cfdi_document_id) +# +# def test_cancel_payment_cfdi(self): +# # Create a payment with related CFDI +# self.payment.move_type = "entry" +# self.payment.is_reconciled = True +# self.payment.create_payment_cfdi() +# +# # Cancel the payment CFDI +# self.payment.cancel_payment_cfdi() +# +# # Check if the CFDI is canceled +# self.assertEqual(self.payment.cfdi_document_id.state, "canceled") diff --git a/l10n_mx_cfdi_account/tests/test_account_tax.py b/l10n_mx_cfdi_account/tests/test_account_tax.py new file mode 100644 index 0000000..560c6ea --- /dev/null +++ b/l10n_mx_cfdi_account/tests/test_account_tax.py @@ -0,0 +1,62 @@ +from odoo.tests import TransactionCase + + +class TestAccountTax(TransactionCase): + def setUp(self): + super().setUp() + self.tax_group = self.env["account.tax.group"].create( + { + "name": "IVA", + "sequence": 1, + } + ) + + self.tax_isr = self.env["account.tax"].create( + { + "name": "ISR Tax", + "amount": 10.0, + "amount_type": "percent", + "type_tax_use": "sale", + "tax_group_id": self.tax_group.id, + } + ) + self.tax_iva = self.env["account.tax"].create( + { + "name": "IVA Tax", + "amount": 16.0, + "amount_type": "percent", + "type_tax_use": "sale", + "tax_group_id": self.tax_group.id, + } + ) + self.tax_ieps = self.env["account.tax"].create( + { + "name": "IEPS Tax", + "amount": 8.0, + "amount_type": "percent", + "type_tax_use": "sale", + "tax_group_id": self.tax_group.id, + } + ) + + def test_extract_l10n_mx_tax_code(self): + self.assertEqual(self.tax_isr.extract_l10n_mx_tax_code(), "ISR") + self.assertEqual(self.tax_iva.extract_l10n_mx_tax_code(), "IVA") + self.assertEqual(self.tax_ieps.extract_l10n_mx_tax_code(), "IEPS") + + tax_without_code = self.env["account.tax"].create( + { + "name": "Test Tax", + "amount": 5.0, + "amount_type": "percent", + "type_tax_use": "sale", + "tax_group_id": self.tax_group.id, + } + ) + + tax_without_code.extract_l10n_mx_tax_code() + + def test_extract_is_retention(self): + self.assertFalse(self.tax_isr.extract_is_retention()) + self.assertFalse(self.tax_iva.extract_is_retention()) + self.assertFalse(self.tax_ieps.extract_is_retention()) diff --git a/l10n_mx_cfdi_account/views/account_move.xml b/l10n_mx_cfdi_account/views/account_move.xml new file mode 100644 index 0000000..303a9c6 --- /dev/null +++ b/l10n_mx_cfdi_account/views/account_move.xml @@ -0,0 +1,139 @@ + + + + account.move.view.form.inherit.cfdi.account + account.move + + + + Send + + + Send + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + account.out.invoice.tree.cfdi + account.move + + + + + + + + + + + account.invoice.filter.cfdi + account.move + + + + + + + + + diff --git a/l10n_mx_cfdi_account/views/account_payment.xml b/l10n_mx_cfdi_account/views/account_payment.xml new file mode 100644 index 0000000..89ab80e --- /dev/null +++ b/l10n_mx_cfdi_account/views/account_payment.xml @@ -0,0 +1,42 @@ + + + Account Payment: Payment Form + account.payment + + + + + + + + + + + + + diff --git a/l10n_mx_cfdi_account/views/account_payment_register.xml b/l10n_mx_cfdi_account/views/account_payment_register.xml new file mode 100644 index 0000000..187c307 --- /dev/null +++ b/l10n_mx_cfdi_account/views/account_payment_register.xml @@ -0,0 +1,12 @@ + + + Account Move Payment: CFDI generation + account.payment.register + + + + + + + + diff --git a/l10n_mx_cfdi_account/views/res_config_settings.xml b/l10n_mx_cfdi_account/views/res_config_settings.xml new file mode 100644 index 0000000..8f861ac --- /dev/null +++ b/l10n_mx_cfdi_account/views/res_config_settings.xml @@ -0,0 +1,22 @@ + + + res.config.settings.view.form.inherit.l10n.mx + res.config.settings + + + + 1 + + + + + + + + + + + diff --git a/l10n_mx_cfdi_account/wizards/__init__.py b/l10n_mx_cfdi_account/wizards/__init__.py new file mode 100644 index 0000000..4d0db3a --- /dev/null +++ b/l10n_mx_cfdi_account/wizards/__init__.py @@ -0,0 +1,3 @@ +from . import document_cancel +from . import create_cfdi_publico_en_general +from . import download_cfdi_files diff --git a/l10n_mx_cfdi_account/wizards/account_invoice_send_views.xml b/l10n_mx_cfdi_account/wizards/account_invoice_send_views.xml new file mode 100644 index 0000000..2ca9d9c --- /dev/null +++ b/l10n_mx_cfdi_account/wizards/account_invoice_send_views.xml @@ -0,0 +1,15 @@ + + + account.move.send.no_print + account.move.send + + +
+ 1 +
+
+ 1 +
+
+
+
diff --git a/l10n_mx_cfdi_account/wizards/create_cfdi_publico_en_general.py b/l10n_mx_cfdi_account/wizards/create_cfdi_publico_en_general.py new file mode 100644 index 0000000..ffa04cb --- /dev/null +++ b/l10n_mx_cfdi_account/wizards/create_cfdi_publico_en_general.py @@ -0,0 +1,166 @@ +from odoo import _, api, fields, models +from odoo.exceptions import ValidationError +from odoo.tools.safe_eval import datetime + + +class CFDIGenericInvoiceCreate(models.TransientModel): + _name = "l10n_mx_cfdi_account.generic_invoice_create" + _description = "Create a generic CFDI invoice" + + periodicity_id = fields.Many2one( + "l10n_mx_catalogs.c_periodicidad", string="Periodicity", required=True + ) + + meses_id = fields.Many2one("l10n_mx_catalogs.c_meses", string="Mes", required=True) + + year = fields.Text(string="Año", required=True) + + issuer_id = fields.Many2one( + "l10n_mx_cfdi.issuer", + string="Emisor", + domain=[("registered", "=", True)], + required=True, + ) + + payment_method_id = fields.Many2one( + "l10n_mx_catalogs.c_metodo_pago", string="Metodo de Pago", readonly=True + ) + payment_form_id = fields.Many2one( + "l10n_mx_catalogs.c_forma_pago", string="Forma de Pago" + ) + + fiscal_regime_id = fields.Many2one( + "l10n_mx_catalogs.c_regimen_fiscal", + string="Régimen Fiscal", + required=True, + compute="_compute_fiscal_regime_id", + readonly=True, + ) + + cfdi_use_id = fields.Many2one( + "l10n_mx_catalogs.c_uso_cfdi", + string="Uso de CFDI", + required=True, + readonly=True, + ) + move_ids = fields.Many2many("account.move", string="Facturas", required=True) + + date = fields.Date(string="Fecha", required=True, default=fields.Date.context_today) + + @api.depends("periodicity_id") + def _compute_fiscal_regime_id(self): + for record in self: + if record.periodicity_id.code == "05": + record.fiscal_regime_id = self.env.ref( + "l10n_mx_catalogs.c_regimen_fiscal_621" + ) + else: + record.fiscal_regime_id = self.env.ref( + "l10n_mx_catalogs.c_regimen_fiscal_616" + ) + + @api.model + def default_get(self, field_names): + defaults_dict = super().default_get(field_names) + context = self.env.context + + if context["active_model"] == "account.move": + related_invoice_objs = self.env["account.move"].browse( + context["active_ids"] + ) + for invoice in related_invoice_objs: + self._validate_invoice(invoice) + + defaults_dict.update({"move_ids": related_invoice_objs}) + + currentDateTime = datetime.datetime.now() + defaults_dict.update( + { + "year": currentDateTime.strftime("%Y"), + "cfdi_use_id": self.env.ref("l10n_mx_catalogs.c_uso_cfdi_S01").id, + "payment_method_id": self.env.ref( + "l10n_mx_catalogs.c_metodo_pago_PUE" + ).id, + "payment_form_id": self.env.ref("l10n_mx_catalogs.c_forma_pago_01").id, + } + ) + return defaults_dict + + @api.constrains("move_ids") + def _validate_included_invoices(self): + for record in self: + for invoice in record.move_ids: + self._validate_invoice(invoice) + + @api.model + def _validate_invoice(self, invoice): + invoice.ensure_one() + + if invoice.state != "posted": + raise ValidationError(_("Invoice %s is not posted.") % invoice.name) + + related_cfdi = invoice.related_cert_ids.filtered_domain( + [("state", "=", "published")] + ) + if related_cfdi: + raise ValidationError( + _("Invoice %s already has a published CFDI.") % invoice.name + ) + + err_msg = invoice.validate_invoice_items_for_cfdi_generation() + if err_msg: + raise ValidationError(err_msg) + + def create_cfdi(self): + """Emit CFDI 'Al Público en General'""" + + self.ensure_one() + receiver = self.env.ref( + "l10n_mx_cfdi.l10n_mx_cfdi_res_partner_publico_en_general" + ) + cert = self.env["l10n_mx_cfdi.document"].create( + { + "type": "I", + "issuer_id": self.issuer_id.id, + "receiver_id": receiver.id, + "is_global_note": True, + } + ) + + try: + all_items_data = [] + for invoice in self.move_ids: + items_data = invoice.gather_invoice_cfdi_items_data() + all_items_data.extend(items_data) + currency = self.move_ids[0].currency_id + + cfdi_data = { + "Currency": currency[0].name, + "ExpeditionPlace": self.issuer_id.zip, + "CfdiType": "I", + "Date": self.move_ids._format_cfdi_date_str(self.date), + "PaymentForm": self.payment_form_id.code, + "PaymentMethod": self.payment_method_id.code, + "GlobalInformation": { + "Periodicity": self.periodicity_id.code, + "Months": self.meses_id.code, + "Year": self.year, + }, + "Receiver": { + "Name": receiver.name, + "Rfc": receiver.vat, + "CfdiUse": self.cfdi_use_id.code, + "FiscalRegime": receiver.tax_regime.code, + "TaxZipCode": self.issuer_id.zip, + }, + "Items": all_items_data, + } + + cert.publish(cfdi_data) + + for invoice in self.move_ids: + invoice.related_cert_ids = [(4, cert.id)] + + except Exception as e: + cert.unlink() + raise e diff --git a/l10n_mx_cfdi_account/wizards/create_cfdi_publico_en_general.xml b/l10n_mx_cfdi_account/wizards/create_cfdi_publico_en_general.xml new file mode 100644 index 0000000..942d691 --- /dev/null +++ b/l10n_mx_cfdi_account/wizards/create_cfdi_publico_en_general.xml @@ -0,0 +1,64 @@ + + + Crear CFDI al Público en General + l10n_mx_cfdi_account.generic_invoice_create + +
+ + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+
+ + + Crear CFDI al Público en General + l10n_mx_cfdi_account.generic_invoice_create + form + new + + list + +
diff --git a/l10n_mx_cfdi_account/wizards/document_cancel.py b/l10n_mx_cfdi_account/wizards/document_cancel.py new file mode 100644 index 0000000..deb0608 --- /dev/null +++ b/l10n_mx_cfdi_account/wizards/document_cancel.py @@ -0,0 +1,77 @@ +from odoo import api, fields, models + + +class CertificateCancel(models.TransientModel): + _name = "l10n_mx_cfdi_account.document_cancel" + _description = "Certificate Cancel" + + certificate_ids = fields.Many2many( + "l10n_mx_cfdi.document", string="Certificados", required=True + ) + cancel_reason_id = fields.Many2one( + "l10n_mx_catalogs.c_motivo_cancelacion", string="Razón", required=True + ) + replacement_certificate_id = fields.Many2one( + "l10n_mx_cfdi.document", string="Reemplazo" + ) + single_cancel = fields.Boolean(default=True) + + related_invoices = fields.Many2many("account.move", string="Facturas Relacionadas") + + requires_replacement = fields.Boolean( + compute="_compute_requires_replacement", store=False + ) + simulate_operation = fields.Boolean( + default=False, + help="Simulate the cancel operation without sending the request to the SAT", + groups="base.group_system", + ) + + @api.depends("cancel_reason_id") + def _compute_requires_replacement(self): + for record in self: + record.requires_replacement = record.cancel_reason_id.code == "01" + + @api.model + def default_get(self, field_names): + defaults_dict = super().default_get(field_names) + context = self.env.context + + if context["active_model"] == "account.move": + related_invoice_objs = self.env["account.move"].browse( + context["active_ids"] + ) + defaults_dict.update( + { + "certificate_ids": ( + related_invoice_objs.related_cert_ids.filtered_domain( + [("state", "=", "published")] + ) + ), + "related_invoices": related_invoice_objs, + "cancel_reason_id": self.env.ref( + "l10n_mx_catalogs.c_motivo_cancelacion_02" + ).id, + } + ) + + return defaults_dict + + def cancel_certificate(self): + for record in self: + for certificate in record.certificate_ids: + if certificate.state == "published": + certificate.cancel( + record.cancel_reason_id.code, + record.replacement_certificate_id, + record.simulate_operation, + ) + + for invoice in record.certificate_ids.related_invoice_id: + invoice._compute_cfdi_document_id() + + if self.env.company.l10n_mx_cfdi_auto: + invoice.button_draft() + + for payment in record.certificate_ids.related_payment_id: + payment.move_id._compute_cfdi_document_id() diff --git a/l10n_mx_cfdi_account/wizards/document_cancel_form.xml b/l10n_mx_cfdi_account/wizards/document_cancel_form.xml new file mode 100644 index 0000000..419b497 --- /dev/null +++ b/l10n_mx_cfdi_account/wizards/document_cancel_form.xml @@ -0,0 +1,42 @@ + + + Cancelar CFDI + l10n_mx_cfdi_account.document_cancel + +
+ + + + + + + + + + + + + +
+
+
+
+
+ + + Cancelar CFDI + l10n_mx_cfdi_account.document_cancel + form + new + +
diff --git a/l10n_mx_cfdi_account/wizards/download_cfdi_files.py b/l10n_mx_cfdi_account/wizards/download_cfdi_files.py new file mode 100644 index 0000000..45ee93f --- /dev/null +++ b/l10n_mx_cfdi_account/wizards/download_cfdi_files.py @@ -0,0 +1,79 @@ +import base64 +import io +import zipfile + +from odoo import api, fields, models + + +class DownloadCFDIFilesWizard(models.TransientModel): + _name = "l10n_mx_cfdi_account.dl_cfdi_files_wizard" + _description = "Create ZIP file containing selected invoices CFDI" + + invoice_ids = fields.Many2many("account.move", string="Facturas", required=True) + cfdi_document_ids = fields.Many2many( + "l10n_mx_cfdi.document", + string="CFDI Documents", + required=True, + relation="l10n_mx_cfdi_download_cfdi_files_wizard_rel", + ) + + zip_file = fields.Many2one("ir.attachment", readonly=True, ondelete="cascade") + + @api.model + def default_get(self, field_names): + defaults_dict = super().default_get(field_names) + context = self.env.context + + if context["active_model"] == "account.move": + related_invoice_objs = self.env["account.move"].browse( + context["active_ids"] + ) + defaults_dict.update( + { + "invoice_ids": related_invoice_objs, + "cfdi_document_ids": related_invoice_objs.mapped( + "cfdi_document_id" + ), + } + ) + + return defaults_dict + + def _create_zip_file(self): + # prepare zip file + stream = io.BytesIO() + zip_archive = zipfile.ZipFile(stream, "w") + + # add docs to zip file + for cfdi_doc in self.cfdi_document_ids: + if cfdi_doc: + cfdi_doc.download_files_if_needed() + + zip_archive.writestr( + cfdi_doc.pdf_filename, base64.b64decode(cfdi_doc.pdf_file) + ) + zip_archive.writestr( + cfdi_doc.xml_filename, base64.b64decode(cfdi_doc.xml_file) + ) + + zip_archive.close() + + bytes_of_zipfile = stream.getvalue() + + # create attachment + self.zip_file = self.env["ir.attachment"].create( + { + "name": "cfdis.zip", + "datas": base64.b64encode(bytes_of_zipfile), + "type": "binary", + } + ) + + def action_download_zip(self): + self._create_zip_file() + + return { + "type": "ir.actions.act_url", + "url": "/web/content/%s?download=true" % self.zip_file.id, + "target": "self", + } diff --git a/l10n_mx_cfdi_account/wizards/download_cfdi_files_wizard.xml b/l10n_mx_cfdi_account/wizards/download_cfdi_files_wizard.xml new file mode 100644 index 0000000..51693ef --- /dev/null +++ b/l10n_mx_cfdi_account/wizards/download_cfdi_files_wizard.xml @@ -0,0 +1,42 @@ + + + + Descargar CFDIs + l10n_mx_cfdi_account.dl_cfdi_files_wizard + +
+

CFDIs Encontrados

+ + + + + + + + + + + +
+
+ +
+
+ + + Descargar CFDIs + l10n_mx_cfdi_account.dl_cfdi_files_wizard + form + new + + list + + +