mirror of
https://github.com/OCA/account-financial-tools.git
synced 2025-02-02 12:47:26 +02:00
[ADD] purchase_unreconciled
This commit is contained in:
committed by
JasminSForgeFlow
parent
2af6e46c54
commit
a571f061ee
93
purchase_unreconciled/README.rst
Normal file
93
purchase_unreconciled/README.rst
Normal file
@@ -0,0 +1,93 @@
|
||||
=====================
|
||||
Purchase Unreconciled
|
||||
=====================
|
||||
|
||||
.. !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||
!! This file is generated by oca-gen-addon-readme !!
|
||||
!! changes will be overwritten. !!
|
||||
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||
|
||||
.. |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-AGPL--3-blue.png
|
||||
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
|
||||
:alt: License: AGPL-3
|
||||
.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Faccount--financial--tools-lightgray.png?logo=github
|
||||
:target: https://github.com/OCA/account-financial-tools/tree/14.0/purchase_unreconciled
|
||||
:alt: OCA/account-financial-tools
|
||||
.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png
|
||||
:target: https://translation.odoo-community.org/projects/account-financial-tools-14-0/account-financial-tools-14-0-purchase_unreconciled
|
||||
:alt: Translate me on Weblate
|
||||
.. |badge5| image:: https://img.shields.io/badge/runbot-Try%20me-875A7B.png
|
||||
:target: https://runbot.odoo-community.org/runbot/92/14.0
|
||||
:alt: Try me on Runbot
|
||||
|
||||
|badge1| |badge2| |badge3| |badge4| |badge5|
|
||||
|
||||
This module adds a new fields "Unreconciled" on Purchase Orders, that allows
|
||||
to find PO's with unreconciled journal items related.
|
||||
|
||||
This module allows to reconcile those PO in a single click. In accounting
|
||||
settings users will be able to set up a specific account for write-off.
|
||||
|
||||
**Table of contents**
|
||||
|
||||
.. contents::
|
||||
:local:
|
||||
|
||||
Usage
|
||||
=====
|
||||
|
||||
Accountants will be able to find a filters in Purchase Orders that shows
|
||||
outstanding balances in interim accounts. Also there is a link in the PO
|
||||
to those outstanding journal items.
|
||||
|
||||
Locking the PO will automatically reconcile the outstanding balance for the
|
||||
stock iterim accounts.
|
||||
|
||||
Bug Tracker
|
||||
===========
|
||||
|
||||
Bugs are tracked on `GitHub Issues <https://github.com/OCA/account-financial-tools/issues>`_.
|
||||
In case of trouble, please check there if your issue has already been reported.
|
||||
If you spotted it first, help us smashing it by providing a detailed and welcomed
|
||||
`feedback <https://github.com/OCA/account-financial-tools/issues/new?body=module:%20purchase_unreconciled%0Aversion:%2014.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_.
|
||||
|
||||
Do not contact contributors directly about support or help with technical issues.
|
||||
|
||||
Credits
|
||||
=======
|
||||
|
||||
Authors
|
||||
~~~~~~~
|
||||
|
||||
* ForgeFlow S.L.
|
||||
|
||||
Contributors
|
||||
~~~~~~~~~~~~
|
||||
|
||||
* ForgeFlow S.L. <contact@forgeflow.com>
|
||||
|
||||
- Lois Rilo <lois.rilo@forgeflow.com>
|
||||
|
||||
- Aaron Henriquez <ahenriquez@forgeflow.com>
|
||||
|
||||
- Miquel Raich <miquel.raich@forgeflow.com>
|
||||
|
||||
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/account-financial-tools <https://github.com/OCA/account-financial-tools/tree/14.0/purchase_unreconciled>`_ project on GitHub.
|
||||
|
||||
You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.
|
||||
1
purchase_unreconciled/__init__.py
Normal file
1
purchase_unreconciled/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
from . import models
|
||||
17
purchase_unreconciled/__manifest__.py
Normal file
17
purchase_unreconciled/__manifest__.py
Normal file
@@ -0,0 +1,17 @@
|
||||
# Copyright 2019 ForgeFlow S.L.
|
||||
# - Lois Rilo Antelo
|
||||
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
{
|
||||
"name": "Purchase Unreconciled",
|
||||
"version": "14.0.1.0.0",
|
||||
"author": "ForgeFlow S.L., Odoo Community Association (OCA)",
|
||||
"website": "https://github.com/OCA/account-financial-tools",
|
||||
"category": "Purchases",
|
||||
"depends": ["account_move_line_purchase_info", "purchase_stock"],
|
||||
"data": ["views/purchase_order_view.xml", "views/res_config_settings_view.xml"],
|
||||
"license": "AGPL-3",
|
||||
"installable": True,
|
||||
"development_status": "Alpha",
|
||||
"maintainers": ["AaronHForgeFlow"],
|
||||
}
|
||||
4
purchase_unreconciled/models/__init__.py
Normal file
4
purchase_unreconciled/models/__init__.py
Normal file
@@ -0,0 +1,4 @@
|
||||
from . import purchase_order
|
||||
from . import company
|
||||
from . import res_config_settings
|
||||
from . import account_move_line
|
||||
72
purchase_unreconciled/models/account_move_line.py
Normal file
72
purchase_unreconciled/models/account_move_line.py
Normal file
@@ -0,0 +1,72 @@
|
||||
# Copyright 2019 ForgeFlow S.L.
|
||||
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
from datetime import datetime
|
||||
|
||||
from odoo import _, models
|
||||
from odoo.exceptions import ValidationError
|
||||
|
||||
|
||||
class AccountMoveLine(models.Model):
|
||||
_inherit = "account.move.line"
|
||||
|
||||
def _get_writeoff_amounts(self):
|
||||
precision = self.env["decimal.precision"].precision_get("Account")
|
||||
writeoff_amount = round(
|
||||
sum([line["amount_residual"] for line in self]), precision
|
||||
)
|
||||
writeoff_amount_curr = round(
|
||||
sum([line["amount_residual_currency"] for line in self]), precision
|
||||
)
|
||||
|
||||
first_currency = self[0]["currency_id"]
|
||||
if all([line["currency_id"] == first_currency for line in self]):
|
||||
same_curr = True
|
||||
else:
|
||||
same_curr = False
|
||||
|
||||
return (
|
||||
writeoff_amount,
|
||||
writeoff_amount_curr,
|
||||
same_curr,
|
||||
)
|
||||
|
||||
def _create_writeoff(self, writeoff_vals):
|
||||
(
|
||||
amount_writeoff,
|
||||
amount_writeoff_curr,
|
||||
same_curr,
|
||||
) = self._get_writeoff_amounts()
|
||||
partners = self.mapped("partner_id")
|
||||
write_off_vals = {
|
||||
"name": _("Automatic writeoff"),
|
||||
"amount_currency": same_curr and amount_writeoff_curr or amount_writeoff,
|
||||
"debit": amount_writeoff > 0.0 and amount_writeoff or 0.0,
|
||||
"credit": amount_writeoff < 0.0 and -amount_writeoff or 0.0,
|
||||
"partner_id": len(partners) == 1 and partners.id or False,
|
||||
"account_id": writeoff_vals["account_id"],
|
||||
"purchase_order_id": writeoff_vals["purchase_order_id"],
|
||||
"journal_id": writeoff_vals["journal_id"],
|
||||
"currency_id": writeoff_vals["currency_id"],
|
||||
}
|
||||
counterpart_account = self.mapped("account_id")
|
||||
if len(counterpart_account) != 1:
|
||||
raise ValidationError(_("CAnnot write-off more than one account"))
|
||||
counter_part = write_off_vals.copy()
|
||||
counter_part["debit"] = write_off_vals["credit"]
|
||||
counter_part["credit"] = write_off_vals["debit"]
|
||||
counter_part["amount_currency"] = -write_off_vals["amount_currency"]
|
||||
counter_part["account_id"] = (counterpart_account.id,)
|
||||
|
||||
move = self.env["account.move"].create(
|
||||
{
|
||||
"date": datetime.now(),
|
||||
"journal_id": writeoff_vals["journal_id"],
|
||||
"currency_id": writeoff_vals["currency_id"],
|
||||
"line_ids": [(0, 0, write_off_vals), (0, 0, counter_part)],
|
||||
}
|
||||
)
|
||||
move.action_post()
|
||||
return move.line_ids.filtered(
|
||||
lambda l: l.account_id.id == counterpart_account.id
|
||||
)
|
||||
20
purchase_unreconciled/models/company.py
Normal file
20
purchase_unreconciled/models/company.py
Normal file
@@ -0,0 +1,20 @@
|
||||
# Copyright 2019 ForgeFlow S.L.
|
||||
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class ResCompany(models.Model):
|
||||
_inherit = "res.company"
|
||||
|
||||
purchase_reconcile_account_id = fields.Many2one(
|
||||
"account.account",
|
||||
domain=lambda self: [("deprecated", "=", False)],
|
||||
string="Write-Off Account On Purchases",
|
||||
ondelete="restrict",
|
||||
copy=False,
|
||||
help="Write-off account to reconcile Unreconciled Purchase Orders",
|
||||
)
|
||||
purchase_reconcile_journal_id = fields.Many2one(
|
||||
"account.journal", string="WriteOff Journal for Purchases"
|
||||
)
|
||||
157
purchase_unreconciled/models/purchase_order.py
Normal file
157
purchase_unreconciled/models/purchase_order.py
Normal file
@@ -0,0 +1,157 @@
|
||||
# Copyright 2019 ForgeFlow S.L.
|
||||
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
from odoo import _, api, exceptions, fields, models
|
||||
from odoo.osv import expression
|
||||
|
||||
|
||||
class PurchaseOrder(models.Model):
|
||||
_inherit = "purchase.order"
|
||||
|
||||
unreconciled = fields.Boolean(
|
||||
compute="_compute_unreconciled",
|
||||
search="_search_unreconciled",
|
||||
help="Indicates that a Purchase Order has related Journal items not "
|
||||
"reconciled.\nNote that if it is false it can be either that "
|
||||
"everything is reconciled or that the related accounts do not "
|
||||
"allow reconciliation",
|
||||
)
|
||||
is_shipped = fields.Boolean(search="_search_is_shipped")
|
||||
|
||||
@api.model
|
||||
def _get_purchase_unreconciled_base_domain(self):
|
||||
included_accounts = (
|
||||
(
|
||||
self.env["product.category"].search(
|
||||
[("property_valuation", "=", "real_time")]
|
||||
)
|
||||
)
|
||||
.mapped("property_stock_account_input_categ_id")
|
||||
.ids
|
||||
)
|
||||
unreconciled_domain = [
|
||||
("account_id.reconcile", "=", True),
|
||||
("account_id", "in", included_accounts),
|
||||
("move_id.state", "=", "posted"),
|
||||
# for some reason when amount_residual is zero
|
||||
# is marked as reconciled, this is better check
|
||||
("full_reconcile_id", "=", False),
|
||||
("company_id", "in", self.env.companies.ids),
|
||||
]
|
||||
return unreconciled_domain
|
||||
|
||||
def _search_is_shipped(self, operator, value):
|
||||
if operator != "=" or not isinstance(value, bool):
|
||||
raise ValueError(_("Unsupported search operator"))
|
||||
is_shipped_pos = self.search([("picking_ids.state", "in", ("done", "cancel"))])
|
||||
if value:
|
||||
return [("id", "in", is_shipped_pos.ids)]
|
||||
else:
|
||||
return [("id", "not in", is_shipped_pos.ids)]
|
||||
|
||||
def _compute_unreconciled(self):
|
||||
acc_item = self.env["account.move.line"]
|
||||
for rec in self:
|
||||
domain = rec._get_purchase_unreconciled_base_domain()
|
||||
unreconciled_domain = expression.AND(
|
||||
[domain, [("purchase_order_id", "=", rec.id)]]
|
||||
)
|
||||
unreconciled_items = acc_item.search(unreconciled_domain)
|
||||
rec.unreconciled = len(unreconciled_items) > 0
|
||||
|
||||
def _search_unreconciled(self, operator, value):
|
||||
if operator != "=" or not isinstance(value, bool):
|
||||
raise ValueError(_("Unsupported search operator"))
|
||||
acc_item = self.env["account.move.line"]
|
||||
domain = self._get_purchase_unreconciled_base_domain()
|
||||
unreconciled_domain = expression.AND(
|
||||
[domain, [("purchase_order_id", "!=", False)]]
|
||||
)
|
||||
unreconciled_items = acc_item.search(unreconciled_domain)
|
||||
unreconciled_pos = unreconciled_items.mapped("purchase_order_id")
|
||||
if value:
|
||||
return [("id", "in", unreconciled_pos.ids)]
|
||||
else:
|
||||
return [("id", "not in", unreconciled_pos.ids)]
|
||||
|
||||
def action_view_unreconciled(self):
|
||||
self.ensure_one()
|
||||
acc_item = self.env["account.move.line"]
|
||||
domain = self._get_purchase_unreconciled_base_domain()
|
||||
unreconciled_domain = expression.AND(
|
||||
[domain, [("purchase_order_id", "=", self.id)]]
|
||||
)
|
||||
unreconciled_items = acc_item.search(unreconciled_domain)
|
||||
action = self.env.ref("account.action_account_moves_all")
|
||||
action_dict = action.read()[0]
|
||||
action_dict["domain"] = [("id", "in", unreconciled_items.ids)]
|
||||
return action_dict
|
||||
|
||||
def action_reconcile(self):
|
||||
if (
|
||||
not self.company_id.purchase_reconcile_account_id
|
||||
or not self.company_id.purchase_reconcile_journal_id
|
||||
):
|
||||
raise exceptions.ValidationError(
|
||||
_(
|
||||
"The write-off account and jounral for purchases is missing. An "
|
||||
"accountant must fill that information"
|
||||
)
|
||||
)
|
||||
self.ensure_one()
|
||||
domain = self._get_purchase_unreconciled_base_domain()
|
||||
unreconciled_domain = expression.AND(
|
||||
[domain, [("purchase_order_id", "=", self.id)]]
|
||||
)
|
||||
unreconciled_domain = expression.AND(
|
||||
[unreconciled_domain, [("company_id", "=", self.company_id.id)]]
|
||||
)
|
||||
unreconciled_items = self.env["account.move.line"].search(unreconciled_domain)
|
||||
writeoff_to_reconcile = False
|
||||
for account in unreconciled_items.mapped("account_id"):
|
||||
acc_unrec_items = unreconciled_items.filtered(
|
||||
lambda ml: ml.account_id == account
|
||||
)
|
||||
all_aml_share_same_currency = all(
|
||||
[x.currency_id == self[0].currency_id for x in acc_unrec_items]
|
||||
)
|
||||
writeoff_vals = {
|
||||
"account_id": self.company_id.purchase_reconcile_account_id.id,
|
||||
"journal_id": self.company_id.purchase_reconcile_journal_id.id,
|
||||
"purchase_order_id": self.id,
|
||||
"currency_id": self.currency_id.id,
|
||||
}
|
||||
if not all_aml_share_same_currency:
|
||||
writeoff_vals["amount_currency"] = False
|
||||
if writeoff_to_reconcile:
|
||||
writeoff_to_reconcile += unreconciled_items._create_writeoff(
|
||||
writeoff_vals
|
||||
)
|
||||
else:
|
||||
writeoff_to_reconcile = unreconciled_items._create_writeoff(
|
||||
writeoff_vals
|
||||
)
|
||||
# add writeoff line to reconcile algorithm and finish the reconciliation
|
||||
if writeoff_to_reconcile:
|
||||
remaining_moves = unreconciled_items + writeoff_to_reconcile
|
||||
else:
|
||||
remaining_moves = unreconciled_items
|
||||
# Check if reconciliation is total or needs an exchange rate entry to be created
|
||||
if remaining_moves:
|
||||
remaining_moves.filtered(lambda l: not l.reconciled).reconcile()
|
||||
return {
|
||||
"name": _("Reconciled journal items"),
|
||||
"type": "ir.actions.act_window",
|
||||
"view_type": "form",
|
||||
"view_mode": "tree,form",
|
||||
"res_model": "account.move.line",
|
||||
"domain": [
|
||||
("id", "in", unreconciled_items.ids + writeoff_to_reconcile.ids)
|
||||
],
|
||||
}
|
||||
|
||||
def button_done(self):
|
||||
for rec in self:
|
||||
if rec.unreconciled:
|
||||
rec.action_reconcile()
|
||||
return super(PurchaseOrder, self).button_done()
|
||||
16
purchase_unreconciled/models/res_config_settings.py
Normal file
16
purchase_unreconciled/models/res_config_settings.py
Normal file
@@ -0,0 +1,16 @@
|
||||
# Copyright 2016 Akretion (Alexis de Lattre <alexis.delattre@akretion.com>)
|
||||
# Copyright 2018 Tecnativa - Pedro M. Baeza
|
||||
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
from odoo import fields, models
|
||||
|
||||
|
||||
class ResConfigSettings(models.TransientModel):
|
||||
_inherit = "res.config.settings"
|
||||
|
||||
purchase_reconcile_account_id = fields.Many2one(
|
||||
related="company_id.purchase_reconcile_account_id", readonly=False
|
||||
)
|
||||
purchase_reconcile_journal_id = fields.Many2one(
|
||||
related="company_id.purchase_reconcile_journal_id", readonly=False
|
||||
)
|
||||
7
purchase_unreconciled/readme/CONTRIBUTORS.rst
Normal file
7
purchase_unreconciled/readme/CONTRIBUTORS.rst
Normal file
@@ -0,0 +1,7 @@
|
||||
* ForgeFlow S.L. <contact@forgeflow.com>
|
||||
|
||||
- Lois Rilo <lois.rilo@forgeflow.com>
|
||||
|
||||
- Aaron Henriquez <ahenriquez@forgeflow.com>
|
||||
|
||||
- Miquel Raich <miquel.raich@forgeflow.com>
|
||||
5
purchase_unreconciled/readme/DESCRIPTION.rst
Normal file
5
purchase_unreconciled/readme/DESCRIPTION.rst
Normal file
@@ -0,0 +1,5 @@
|
||||
This module adds a new fields "Unreconciled" on Purchase Orders, that allows
|
||||
to find PO's with unreconciled journal items related.
|
||||
|
||||
This module allows to reconcile those PO in a single click. In accounting
|
||||
settings users will be able to set up a specific account for write-off.
|
||||
6
purchase_unreconciled/readme/USAGE.rst
Normal file
6
purchase_unreconciled/readme/USAGE.rst
Normal file
@@ -0,0 +1,6 @@
|
||||
Accountants will be able to find a filters in Purchase Orders that shows
|
||||
outstanding balances in interim accounts. Also there is a link in the PO
|
||||
to those outstanding journal items.
|
||||
|
||||
Locking the PO will automatically reconcile the outstanding balance for the
|
||||
stock iterim accounts.
|
||||
436
purchase_unreconciled/static/description/index.html
Normal file
436
purchase_unreconciled/static/description/index.html
Normal file
@@ -0,0 +1,436 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
|
||||
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
|
||||
<head>
|
||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
||||
<meta name="generator" content="Docutils 0.14: http://docutils.sourceforge.net/" />
|
||||
<title>Purchase Unreconciled</title>
|
||||
<style type="text/css">
|
||||
|
||||
/*
|
||||
:Author: David Goodger (goodger@python.org)
|
||||
:Id: $Id: html4css1.css 7952 2016-07-26 18:15:59Z milde $
|
||||
:Copyright: This stylesheet has been placed in the public domain.
|
||||
|
||||
Default cascading style sheet for the HTML output of Docutils.
|
||||
|
||||
See http://docutils.sf.net/docs/howto/html-stylesheets.html for how to
|
||||
customize this style sheet.
|
||||
*/
|
||||
|
||||
/* used to remove borders from tables and images */
|
||||
.borderless, table.borderless td, table.borderless th {
|
||||
border: 0 }
|
||||
|
||||
table.borderless td, table.borderless th {
|
||||
/* Override padding for "table.docutils td" with "! important".
|
||||
The right padding separates the table cells. */
|
||||
padding: 0 0.5em 0 0 ! important }
|
||||
|
||||
.first {
|
||||
/* Override more specific margin styles with "! important". */
|
||||
margin-top: 0 ! important }
|
||||
|
||||
.last, .with-subtitle {
|
||||
margin-bottom: 0 ! important }
|
||||
|
||||
.hidden {
|
||||
display: none }
|
||||
|
||||
.subscript {
|
||||
vertical-align: sub;
|
||||
font-size: smaller }
|
||||
|
||||
.superscript {
|
||||
vertical-align: super;
|
||||
font-size: smaller }
|
||||
|
||||
a.toc-backref {
|
||||
text-decoration: none ;
|
||||
color: black }
|
||||
|
||||
blockquote.epigraph {
|
||||
margin: 2em 5em ; }
|
||||
|
||||
dl.docutils dd {
|
||||
margin-bottom: 0.5em }
|
||||
|
||||
object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] {
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
/* Uncomment (and remove this text!) to get bold-faced definition list terms
|
||||
dl.docutils dt {
|
||||
font-weight: bold }
|
||||
*/
|
||||
|
||||
div.abstract {
|
||||
margin: 2em 5em }
|
||||
|
||||
div.abstract p.topic-title {
|
||||
font-weight: bold ;
|
||||
text-align: center }
|
||||
|
||||
div.admonition, div.attention, div.caution, div.danger, div.error,
|
||||
div.hint, div.important, div.note, div.tip, div.warning {
|
||||
margin: 2em ;
|
||||
border: medium outset ;
|
||||
padding: 1em }
|
||||
|
||||
div.admonition p.admonition-title, div.hint p.admonition-title,
|
||||
div.important p.admonition-title, div.note p.admonition-title,
|
||||
div.tip p.admonition-title {
|
||||
font-weight: bold ;
|
||||
font-family: sans-serif }
|
||||
|
||||
div.attention p.admonition-title, div.caution p.admonition-title,
|
||||
div.danger p.admonition-title, div.error p.admonition-title,
|
||||
div.warning p.admonition-title, .code .error {
|
||||
color: red ;
|
||||
font-weight: bold ;
|
||||
font-family: sans-serif }
|
||||
|
||||
/* Uncomment (and remove this text!) to get reduced vertical space in
|
||||
compound paragraphs.
|
||||
div.compound .compound-first, div.compound .compound-middle {
|
||||
margin-bottom: 0.5em }
|
||||
|
||||
div.compound .compound-last, div.compound .compound-middle {
|
||||
margin-top: 0.5em }
|
||||
*/
|
||||
|
||||
div.dedication {
|
||||
margin: 2em 5em ;
|
||||
text-align: center ;
|
||||
font-style: italic }
|
||||
|
||||
div.dedication p.topic-title {
|
||||
font-weight: bold ;
|
||||
font-style: normal }
|
||||
|
||||
div.figure {
|
||||
margin-left: 2em ;
|
||||
margin-right: 2em }
|
||||
|
||||
div.footer, div.header {
|
||||
clear: both;
|
||||
font-size: smaller }
|
||||
|
||||
div.line-block {
|
||||
display: block ;
|
||||
margin-top: 1em ;
|
||||
margin-bottom: 1em }
|
||||
|
||||
div.line-block div.line-block {
|
||||
margin-top: 0 ;
|
||||
margin-bottom: 0 ;
|
||||
margin-left: 1.5em }
|
||||
|
||||
div.sidebar {
|
||||
margin: 0 0 0.5em 1em ;
|
||||
border: medium outset ;
|
||||
padding: 1em ;
|
||||
background-color: #ffffee ;
|
||||
width: 40% ;
|
||||
float: right ;
|
||||
clear: right }
|
||||
|
||||
div.sidebar p.rubric {
|
||||
font-family: sans-serif ;
|
||||
font-size: medium }
|
||||
|
||||
div.system-messages {
|
||||
margin: 5em }
|
||||
|
||||
div.system-messages h1 {
|
||||
color: red }
|
||||
|
||||
div.system-message {
|
||||
border: medium outset ;
|
||||
padding: 1em }
|
||||
|
||||
div.system-message p.system-message-title {
|
||||
color: red ;
|
||||
font-weight: bold }
|
||||
|
||||
div.topic {
|
||||
margin: 2em }
|
||||
|
||||
h1.section-subtitle, h2.section-subtitle, h3.section-subtitle,
|
||||
h4.section-subtitle, h5.section-subtitle, h6.section-subtitle {
|
||||
margin-top: 0.4em }
|
||||
|
||||
h1.title {
|
||||
text-align: center }
|
||||
|
||||
h2.subtitle {
|
||||
text-align: center }
|
||||
|
||||
hr.docutils {
|
||||
width: 75% }
|
||||
|
||||
img.align-left, .figure.align-left, object.align-left, table.align-left {
|
||||
clear: left ;
|
||||
float: left ;
|
||||
margin-right: 1em }
|
||||
|
||||
img.align-right, .figure.align-right, object.align-right, table.align-right {
|
||||
clear: right ;
|
||||
float: right ;
|
||||
margin-left: 1em }
|
||||
|
||||
img.align-center, .figure.align-center, object.align-center {
|
||||
display: block;
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
table.align-center {
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
.align-left {
|
||||
text-align: left }
|
||||
|
||||
.align-center {
|
||||
clear: both ;
|
||||
text-align: center }
|
||||
|
||||
.align-right {
|
||||
text-align: right }
|
||||
|
||||
/* reset inner alignment in figures */
|
||||
div.align-right {
|
||||
text-align: inherit }
|
||||
|
||||
/* div.align-center * { */
|
||||
/* text-align: left } */
|
||||
|
||||
.align-top {
|
||||
vertical-align: top }
|
||||
|
||||
.align-middle {
|
||||
vertical-align: middle }
|
||||
|
||||
.align-bottom {
|
||||
vertical-align: bottom }
|
||||
|
||||
ol.simple, ul.simple {
|
||||
margin-bottom: 1em }
|
||||
|
||||
ol.arabic {
|
||||
list-style: decimal }
|
||||
|
||||
ol.loweralpha {
|
||||
list-style: lower-alpha }
|
||||
|
||||
ol.upperalpha {
|
||||
list-style: upper-alpha }
|
||||
|
||||
ol.lowerroman {
|
||||
list-style: lower-roman }
|
||||
|
||||
ol.upperroman {
|
||||
list-style: upper-roman }
|
||||
|
||||
p.attribution {
|
||||
text-align: right ;
|
||||
margin-left: 50% }
|
||||
|
||||
p.caption {
|
||||
font-style: italic }
|
||||
|
||||
p.credits {
|
||||
font-style: italic ;
|
||||
font-size: smaller }
|
||||
|
||||
p.label {
|
||||
white-space: nowrap }
|
||||
|
||||
p.rubric {
|
||||
font-weight: bold ;
|
||||
font-size: larger ;
|
||||
color: maroon ;
|
||||
text-align: center }
|
||||
|
||||
p.sidebar-title {
|
||||
font-family: sans-serif ;
|
||||
font-weight: bold ;
|
||||
font-size: larger }
|
||||
|
||||
p.sidebar-subtitle {
|
||||
font-family: sans-serif ;
|
||||
font-weight: bold }
|
||||
|
||||
p.topic-title {
|
||||
font-weight: bold }
|
||||
|
||||
pre.address {
|
||||
margin-bottom: 0 ;
|
||||
margin-top: 0 ;
|
||||
font: inherit }
|
||||
|
||||
pre.literal-block, pre.doctest-block, pre.math, pre.code {
|
||||
margin-left: 2em ;
|
||||
margin-right: 2em }
|
||||
|
||||
pre.code .ln { color: grey; } /* line numbers */
|
||||
pre.code, code { background-color: #eeeeee }
|
||||
pre.code .comment, code .comment { color: #5C6576 }
|
||||
pre.code .keyword, code .keyword { color: #3B0D06; font-weight: bold }
|
||||
pre.code .literal.string, code .literal.string { color: #0C5404 }
|
||||
pre.code .name.builtin, code .name.builtin { color: #352B84 }
|
||||
pre.code .deleted, code .deleted { background-color: #DEB0A1}
|
||||
pre.code .inserted, code .inserted { background-color: #A3D289}
|
||||
|
||||
span.classifier {
|
||||
font-family: sans-serif ;
|
||||
font-style: oblique }
|
||||
|
||||
span.classifier-delimiter {
|
||||
font-family: sans-serif ;
|
||||
font-weight: bold }
|
||||
|
||||
span.interpreted {
|
||||
font-family: sans-serif }
|
||||
|
||||
span.option {
|
||||
white-space: nowrap }
|
||||
|
||||
span.pre {
|
||||
white-space: pre }
|
||||
|
||||
span.problematic {
|
||||
color: red }
|
||||
|
||||
span.section-subtitle {
|
||||
/* font-size relative to parent (h1..h6 element) */
|
||||
font-size: 80% }
|
||||
|
||||
table.citation {
|
||||
border-left: solid 1px gray;
|
||||
margin-left: 1px }
|
||||
|
||||
table.docinfo {
|
||||
margin: 2em 4em }
|
||||
|
||||
table.docutils {
|
||||
margin-top: 0.5em ;
|
||||
margin-bottom: 0.5em }
|
||||
|
||||
table.footnote {
|
||||
border-left: solid 1px black;
|
||||
margin-left: 1px }
|
||||
|
||||
table.docutils td, table.docutils th,
|
||||
table.docinfo td, table.docinfo th {
|
||||
padding-left: 0.5em ;
|
||||
padding-right: 0.5em ;
|
||||
vertical-align: top }
|
||||
|
||||
table.docutils th.field-name, table.docinfo th.docinfo-name {
|
||||
font-weight: bold ;
|
||||
text-align: left ;
|
||||
white-space: nowrap ;
|
||||
padding-left: 0 }
|
||||
|
||||
/* "booktabs" style (no vertical lines) */
|
||||
table.docutils.booktabs {
|
||||
border: 0px;
|
||||
border-top: 2px solid;
|
||||
border-bottom: 2px solid;
|
||||
border-collapse: collapse;
|
||||
}
|
||||
table.docutils.booktabs * {
|
||||
border: 0px;
|
||||
}
|
||||
table.docutils.booktabs th {
|
||||
border-bottom: thin solid;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
h1 tt.docutils, h2 tt.docutils, h3 tt.docutils,
|
||||
h4 tt.docutils, h5 tt.docutils, h6 tt.docutils {
|
||||
font-size: 100% }
|
||||
|
||||
ul.auto-toc {
|
||||
list-style-type: none }
|
||||
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="document" id="purchase-unreconciled">
|
||||
<h1 class="title">Purchase Unreconciled</h1>
|
||||
|
||||
<!-- !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||
!! This file is generated by oca-gen-addon-readme !!
|
||||
!! changes will be overwritten. !!
|
||||
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! -->
|
||||
<p><a class="reference external" href="https://odoo-community.org/page/development-status"><img alt="Beta" src="https://img.shields.io/badge/maturity-Beta-yellow.png" /></a> <a class="reference external" href="http://www.gnu.org/licenses/agpl-3.0-standalone.html"><img alt="License: AGPL-3" src="https://img.shields.io/badge/licence-AGPL--3-blue.png" /></a> <a class="reference external" href="https://github.com/OCA/account-financial-tools/tree/14.0/purchase_unreconciled"><img alt="OCA/account-financial-tools" src="https://img.shields.io/badge/github-OCA%2Faccount--financial--tools-lightgray.png?logo=github" /></a> <a class="reference external" href="https://translation.odoo-community.org/projects/account-financial-tools-14-0/account-financial-tools-14-0-purchase_unreconciled"><img alt="Translate me on Weblate" src="https://img.shields.io/badge/weblate-Translate%20me-F47D42.png" /></a> <a class="reference external" href="https://runbot.odoo-community.org/runbot/92/14.0"><img alt="Try me on Runbot" src="https://img.shields.io/badge/runbot-Try%20me-875A7B.png" /></a></p>
|
||||
<p>This module adds a new fields “Unreconciled” on Purchase Orders, that allows
|
||||
to find PO’s with unreconciled journal items related.</p>
|
||||
<p>This module allows to reconcile those PO in a single click. In accounting
|
||||
settings users will be able to set up a specific account for write-off.</p>
|
||||
<p><strong>Table of contents</strong></p>
|
||||
<div class="contents local topic" id="contents">
|
||||
<ul class="simple">
|
||||
<li><a class="reference internal" href="#usage" id="id1">Usage</a></li>
|
||||
<li><a class="reference internal" href="#bug-tracker" id="id2">Bug Tracker</a></li>
|
||||
<li><a class="reference internal" href="#credits" id="id3">Credits</a><ul>
|
||||
<li><a class="reference internal" href="#authors" id="id4">Authors</a></li>
|
||||
<li><a class="reference internal" href="#contributors" id="id5">Contributors</a></li>
|
||||
<li><a class="reference internal" href="#maintainers" id="id6">Maintainers</a></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="usage">
|
||||
<h1><a class="toc-backref" href="#id1">Usage</a></h1>
|
||||
<p>Accountants will be able to find a filters in Purchase Orders that shows
|
||||
outstanding balances in interim accounts. Also there is a link in the PO
|
||||
to those outstanding journal items.</p>
|
||||
<p>Locking the PO will automatically reconcile the outstanding balance for the
|
||||
stock iterim accounts.</p>
|
||||
</div>
|
||||
<div class="section" id="bug-tracker">
|
||||
<h1><a class="toc-backref" href="#id2">Bug Tracker</a></h1>
|
||||
<p>Bugs are tracked on <a class="reference external" href="https://github.com/OCA/account-financial-tools/issues">GitHub Issues</a>.
|
||||
In case of trouble, please check there if your issue has already been reported.
|
||||
If you spotted it first, help us smashing it by providing a detailed and welcomed
|
||||
<a class="reference external" href="https://github.com/OCA/account-financial-tools/issues/new?body=module:%20purchase_unreconciled%0Aversion:%2014.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**">feedback</a>.</p>
|
||||
<p>Do not contact contributors directly about support or help with technical issues.</p>
|
||||
</div>
|
||||
<div class="section" id="credits">
|
||||
<h1><a class="toc-backref" href="#id3">Credits</a></h1>
|
||||
<div class="section" id="authors">
|
||||
<h2><a class="toc-backref" href="#id4">Authors</a></h2>
|
||||
<ul class="simple">
|
||||
<li>ForgeFlow S.L.</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="contributors">
|
||||
<h2><a class="toc-backref" href="#id5">Contributors</a></h2>
|
||||
<ul class="simple">
|
||||
<li>ForgeFlow S.L. <<a class="reference external" href="mailto:contact@forgeflow.com">contact@forgeflow.com</a>><ul>
|
||||
<li>Lois Rilo <<a class="reference external" href="mailto:lois.rilo@forgeflow.com">lois.rilo@forgeflow.com</a>></li>
|
||||
<li>Aaron Henriquez <<a class="reference external" href="mailto:ahenriquez@forgeflow.com">ahenriquez@forgeflow.com</a>></li>
|
||||
<li>Miquel Raich <<a class="reference external" href="mailto:miquel.raich@forgeflow.com">miquel.raich@forgeflow.com</a>></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="maintainers">
|
||||
<h2><a class="toc-backref" href="#id6">Maintainers</a></h2>
|
||||
<p>This module is maintained by the OCA.</p>
|
||||
<a class="reference external image-reference" href="https://odoo-community.org"><img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" /></a>
|
||||
<p>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.</p>
|
||||
<p>This module is part of the <a class="reference external" href="https://github.com/OCA/account-financial-tools/tree/14.0/purchase_unreconciled">OCA/account-financial-tools</a> project on GitHub.</p>
|
||||
<p>You are welcome to contribute. To learn how please visit <a class="reference external" href="https://odoo-community.org/page/Contribute">https://odoo-community.org/page/Contribute</a>.</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
1
purchase_unreconciled/tests/__init__.py
Normal file
1
purchase_unreconciled/tests/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
from . import test_purchase_unreconciled
|
||||
345
purchase_unreconciled/tests/test_purchase_unreconciled.py
Normal file
345
purchase_unreconciled/tests/test_purchase_unreconciled.py
Normal file
@@ -0,0 +1,345 @@
|
||||
# Copyright 2019 ForgeFlow S.L.
|
||||
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
from datetime import datetime
|
||||
|
||||
from odoo import exceptions, fields
|
||||
from odoo.tests.common import Form, SingleTransactionCase
|
||||
|
||||
|
||||
class TestPurchaseUnreconciled(SingleTransactionCase):
|
||||
@classmethod
|
||||
def setUpClass(cls):
|
||||
super().setUpClass()
|
||||
|
||||
cls.po_obj = cls.env["purchase.order"]
|
||||
cls.product_obj = cls.env["product.product"]
|
||||
cls.category_obj = cls.env["product.category"]
|
||||
cls.partner_obj = cls.env["res.partner"]
|
||||
cls.acc_obj = cls.env["account.account"]
|
||||
cls.invoice_obj = cls.env["account.move"]
|
||||
cls.company = cls.env.ref("base.main_company")
|
||||
cls.company.anglo_saxon_accounting = True
|
||||
assets = cls.env.ref("account.data_account_type_current_assets")
|
||||
expenses = cls.env.ref("account.data_account_type_expenses")
|
||||
equity = cls.env.ref("account.data_account_type_equity")
|
||||
# Create partner:
|
||||
cls.partner = cls.partner_obj.create({"name": "Test Vendor"})
|
||||
# Create product that uses a reconcilable stock input account.
|
||||
cls.account = cls.acc_obj.create(
|
||||
{
|
||||
"name": "Test stock input account",
|
||||
"code": 9999,
|
||||
"user_type_id": assets.id,
|
||||
"reconcile": True,
|
||||
"company_id": cls.company.id,
|
||||
}
|
||||
)
|
||||
cls.writeoff_acc = cls.acc_obj.create(
|
||||
{
|
||||
"name": "Write-offf account",
|
||||
"code": 8888,
|
||||
"user_type_id": expenses.id,
|
||||
"reconcile": True,
|
||||
"company_id": cls.company.id,
|
||||
}
|
||||
)
|
||||
cls.stock_journal = cls.env["account.journal"].create(
|
||||
{"name": "Stock Journal", "code": "STJTEST", "type": "general"}
|
||||
)
|
||||
# Create account for Goods Received Not Invoiced
|
||||
name = "Goods Received Not Invoiced"
|
||||
code = "grni"
|
||||
acc_type = equity
|
||||
cls.account_grni = cls._create_account(
|
||||
acc_type, name, code, cls.company, reconcile=True
|
||||
)
|
||||
# Create account for Cost of Goods Sold
|
||||
name = "Cost of Goods Sold"
|
||||
code = "cogs"
|
||||
acc_type = expenses
|
||||
cls.account_cogs = cls._create_account(acc_type, name, code, cls.company)
|
||||
# Create account for Goods Delivered Not Invoiced
|
||||
name = "Goods Delivered Not Invoiced"
|
||||
code = "gdni"
|
||||
acc_type = expenses
|
||||
cls.account_gdni = cls._create_account(
|
||||
acc_type, name, code, cls.company, reconcile=True
|
||||
)
|
||||
# Create account for Inventory
|
||||
name = "Inventory"
|
||||
code = "inventory"
|
||||
acc_type = assets
|
||||
cls.account_inventory = cls._create_account(acc_type, name, code, cls.company)
|
||||
cls.product_categ = cls.category_obj.create(
|
||||
{
|
||||
"name": "Test Category",
|
||||
"property_cost_method": "standard",
|
||||
"property_stock_valuation_account_id": cls.account_inventory.id,
|
||||
"property_stock_account_input_categ_id": cls.account_grni.id,
|
||||
"property_account_expense_categ_id": cls.account_cogs.id,
|
||||
"property_stock_account_output_categ_id": cls.account_gdni.id,
|
||||
"property_valuation": "real_time",
|
||||
"property_stock_journal": cls.stock_journal.id,
|
||||
}
|
||||
)
|
||||
cls.product_to_reconcile = cls.product_obj.create(
|
||||
{
|
||||
"name": "Purchased Product (To reconcile)",
|
||||
"type": "product",
|
||||
"standard_price": 100.0,
|
||||
"categ_id": cls.product_categ.id,
|
||||
}
|
||||
)
|
||||
|
||||
# Create PO's:
|
||||
cls.po = cls.po_obj.create(
|
||||
{
|
||||
"partner_id": cls.partner.id,
|
||||
"order_line": [
|
||||
(
|
||||
0,
|
||||
0,
|
||||
{
|
||||
"product_id": cls.product_to_reconcile.id,
|
||||
"name": cls.product_to_reconcile.name,
|
||||
"product_qty": 5.0,
|
||||
"price_unit": 100.0,
|
||||
"product_uom": cls.product_to_reconcile.uom_id.id,
|
||||
"date_planned": fields.Datetime.now(),
|
||||
},
|
||||
)
|
||||
],
|
||||
}
|
||||
)
|
||||
cls.po_2 = cls.po_obj.create(
|
||||
{
|
||||
"partner_id": cls.partner.id,
|
||||
"order_line": [
|
||||
(
|
||||
0,
|
||||
0,
|
||||
{
|
||||
"product_id": cls.product_to_reconcile.id,
|
||||
"name": cls.product_to_reconcile.name,
|
||||
"product_qty": 5.0,
|
||||
"price_unit": 100.0,
|
||||
"product_uom": cls.product_to_reconcile.uom_id.id,
|
||||
"date_planned": fields.Datetime.now(),
|
||||
},
|
||||
)
|
||||
],
|
||||
}
|
||||
)
|
||||
# company settings for automated valuation
|
||||
cls.company.purchase_reconcile_account_id = cls.writeoff_acc
|
||||
cls.company.purchase_reconcile_journal_id = cls.stock_journal
|
||||
|
||||
@classmethod
|
||||
def _create_account(cls, acc_type, name, code, company, reconcile=False):
|
||||
"""Create an account."""
|
||||
account = cls.acc_obj.create(
|
||||
{
|
||||
"name": name,
|
||||
"code": code,
|
||||
"user_type_id": acc_type.id,
|
||||
"company_id": company.id,
|
||||
"reconcile": reconcile,
|
||||
}
|
||||
)
|
||||
return account
|
||||
|
||||
def _create_delivery(
|
||||
self,
|
||||
product,
|
||||
qty,
|
||||
):
|
||||
return self.env["stock.picking"].create(
|
||||
{
|
||||
"name": self.product_to_reconcile.name,
|
||||
"partner_id": self.partner.id,
|
||||
"picking_type_id": self.env.ref("stock.picking_type_out").id,
|
||||
"location_id": self.env.ref("stock.stock_location_stock").id,
|
||||
"location_dest_id": self.env.ref("stock.stock_location_customers").id,
|
||||
"move_lines": [
|
||||
(
|
||||
0,
|
||||
0,
|
||||
{
|
||||
"name": self.product_to_reconcile.name,
|
||||
"product_id": self.product_to_reconcile.id,
|
||||
"product_uom": self.product_to_reconcile.uom_id.id,
|
||||
"product_uom_qty": qty,
|
||||
"location_id": self.env.ref(
|
||||
"stock.stock_location_stock"
|
||||
).id,
|
||||
"location_dest_id": self.env.ref(
|
||||
"stock.stock_location_customers"
|
||||
).id,
|
||||
"procure_method": "make_to_stock",
|
||||
},
|
||||
)
|
||||
],
|
||||
}
|
||||
)
|
||||
|
||||
def _do_picking(self, picking, date):
|
||||
"""Do picking with only one move on the given date."""
|
||||
picking.action_confirm()
|
||||
picking.move_lines.quantity_done = picking.move_lines.product_uom_qty
|
||||
picking._action_done()
|
||||
for move in picking.move_lines:
|
||||
move.date = date
|
||||
|
||||
def test_01_nothing_to_reconcile(self):
|
||||
po = self.po
|
||||
self.assertEqual(po.state, "draft")
|
||||
po.button_confirm()
|
||||
self._do_picking(po.picking_ids, fields.Datetime.now())
|
||||
self.assertTrue(po.unreconciled)
|
||||
# Invoice created and validated:
|
||||
po.action_create_invoice()
|
||||
po.invoice_ids.invoice_date = datetime.now()
|
||||
po.invoice_ids.action_post()
|
||||
self.assertEqual(po.state, "purchase")
|
||||
# odoo does it automatically
|
||||
po._compute_unreconciled()
|
||||
self.assertFalse(po.unreconciled)
|
||||
|
||||
def test_03_search_unreconciled(self):
|
||||
"""Test searching unreconciled PO's."""
|
||||
po = self.po_2
|
||||
po.button_confirm()
|
||||
self._do_picking(po.picking_ids, fields.Datetime.now())
|
||||
res = self.po_obj.search([("unreconciled", "=", True)])
|
||||
po._compute_unreconciled()
|
||||
self.assertIn(po, res)
|
||||
self.assertNotIn(self.po, res)
|
||||
# Test value error:
|
||||
with self.assertRaises(ValueError):
|
||||
self.po_obj.search([("unreconciled", "=", "true")])
|
||||
|
||||
def test_04_action_reconcile(self):
|
||||
"""Test reconcile."""
|
||||
# Invoice created and validated:
|
||||
po = self.po_2
|
||||
self.assertTrue(po.unreconciled)
|
||||
po.action_create_invoice()
|
||||
invoice_form = Form(po.invoice_ids[0])
|
||||
# v14 reconciles automatically so here we force discrepancy
|
||||
# with invoice_form.edit(0) as inv_form:
|
||||
invoice_form.invoice_date = datetime.now()
|
||||
with invoice_form.invoice_line_ids.edit(0) as line_form:
|
||||
line_form.price_unit = 99
|
||||
invoice = invoice_form.save()
|
||||
invoice.action_post()
|
||||
self.assertTrue(po.unreconciled)
|
||||
po.action_reconcile()
|
||||
po._compute_unreconciled()
|
||||
self.assertFalse(po.unreconciled)
|
||||
|
||||
def test_05_button_done_reconcile(self):
|
||||
"""Test auto reconcile when locking po."""
|
||||
po = self.po_2.copy()
|
||||
po.company_id.purchase_reconcile_account_id = self.writeoff_acc
|
||||
po.button_confirm()
|
||||
self._do_picking(po.picking_ids, fields.Datetime.now())
|
||||
# Invoice created and validated:
|
||||
# Odoo reconciles automatically so here we force discrepancy
|
||||
po.action_create_invoice()
|
||||
invoice_form = Form(po.invoice_ids[0])
|
||||
invoice_form.invoice_date = datetime.now()
|
||||
with invoice_form.invoice_line_ids.edit(0) as line_form:
|
||||
line_form.price_unit = 99
|
||||
invoice = invoice_form.save()
|
||||
invoice.action_post()
|
||||
self.assertTrue(po.unreconciled)
|
||||
# check error if raised if not write-off account
|
||||
with self.assertRaises(exceptions.ValidationError):
|
||||
self.company.purchase_reconcile_account_id = False
|
||||
po.button_done()
|
||||
# restore the write off account
|
||||
self.company.purchase_reconcile_account_id = self.writeoff_acc
|
||||
po.button_done()
|
||||
po._compute_unreconciled()
|
||||
self.assertFalse(po.unreconciled)
|
||||
|
||||
def test_06_dropship_not_reconcile_sale_journal_items(self):
|
||||
"""
|
||||
Create a fake dropship and lock the PO before receiving the customer
|
||||
invoice. The PO should not close the stock interim output account
|
||||
"""
|
||||
# to create the fake dropship we create a delivery and attach the
|
||||
# journals to the purchase order craeted later
|
||||
self.env["stock.quant"].create(
|
||||
{
|
||||
"product_id": self.product_to_reconcile.id,
|
||||
"location_id": self.env.ref("stock.stock_location_stock").id,
|
||||
"quantity": 1.0,
|
||||
}
|
||||
)
|
||||
delivery = self._create_delivery(self.product_to_reconcile, 1)
|
||||
self._do_picking(delivery, fields.Datetime.now())
|
||||
# We create the PO now and receive it
|
||||
po = self.po_2.copy()
|
||||
po.button_confirm()
|
||||
self._do_picking(po.picking_ids, fields.Datetime.now())
|
||||
self.assertTrue(po.unreconciled)
|
||||
# as long stock_dropshipping is not dependency, I force the PO to be in
|
||||
# the journal items of the delivery
|
||||
delivery_name = delivery.name
|
||||
delivery_ji = self.env["account.move.line"].search(
|
||||
[("move_id.ref", "=", delivery_name)]
|
||||
)
|
||||
delivery_ji.write(
|
||||
{"purchase_line_id": po.order_line[0], "purchase_order_id": po.id}
|
||||
)
|
||||
# then I lock the po to force reconciliation
|
||||
po.button_done()
|
||||
po._compute_unreconciled()
|
||||
self.assertFalse(po.unreconciled)
|
||||
# the PO is reconciled and the stock interim deliverd account is not
|
||||
# reconciled yet
|
||||
for jii in delivery_ji:
|
||||
self.assertFalse(jii.reconciled)
|
||||
|
||||
def test_07_multicompany(self):
|
||||
"""
|
||||
Force the company in the vendor bill to be wrong. The system will
|
||||
write-off the journals for the shipment because those are the only ones
|
||||
with the correct company
|
||||
"""
|
||||
po = self.po.copy()
|
||||
po.button_confirm()
|
||||
self._do_picking(po.picking_ids, fields.Datetime.now())
|
||||
# Invoice created and validated:
|
||||
move_form = Form(self.invoice_obj.with_context(default_type="in_invoice"))
|
||||
move_form.partner_id = self.partner
|
||||
move_form.purchase_id = po
|
||||
invoice = move_form.save()
|
||||
chicago_journal = self.env["account.journal"].create(
|
||||
{
|
||||
"name": "chicago",
|
||||
"code": "ref",
|
||||
"type": "sale",
|
||||
"company_id": self.ref("stock.res_company_1"),
|
||||
}
|
||||
)
|
||||
invoice.write(
|
||||
{
|
||||
"company_id": self.ref("stock.res_company_1"),
|
||||
"journal_id": chicago_journal.id,
|
||||
}
|
||||
)
|
||||
invoice.action_post()
|
||||
self.assertEqual(po.state, "purchase")
|
||||
# The bill is wrong so this is unreconciled
|
||||
self.assertTrue(po.unreconciled)
|
||||
po.button_done()
|
||||
po._compute_unreconciled()
|
||||
self.assertFalse(po.unreconciled)
|
||||
# we check all the journals for the po have the same company
|
||||
ji = self.env["account.move.line"].search(
|
||||
[("purchase_order_id", "=", po.id), ("move_id", "!=", invoice.id)]
|
||||
)
|
||||
self.assertEqual(po.company_id, ji.mapped("company_id"))
|
||||
58
purchase_unreconciled/views/purchase_order_view.xml
Normal file
58
purchase_unreconciled/views/purchase_order_view.xml
Normal file
@@ -0,0 +1,58 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<odoo>
|
||||
<record id="purchase_order_form" model="ir.ui.view">
|
||||
<field name="name">purchase.order.form - purchase_unreconciled</field>
|
||||
<field name="model">purchase.order</field>
|
||||
<field name="inherit_id" ref="purchase.purchase_order_form" />
|
||||
<field name="arch" type="xml">
|
||||
<field name="invoice_status" position="after">
|
||||
<field name="unreconciled" />
|
||||
</field>
|
||||
<div name="button_box" position="inside">
|
||||
<button
|
||||
type="object"
|
||||
string="Unreconciled Journal Items"
|
||||
name="action_view_unreconciled"
|
||||
groups="account.group_account_manager"
|
||||
class="oe_stat_button"
|
||||
icon="fa-gears"
|
||||
attrs="{'invisible':[('unreconciled', '=', False)]}"
|
||||
>
|
||||
</button>
|
||||
<button
|
||||
type="object"
|
||||
string="Reconcile"
|
||||
name="action_reconcile"
|
||||
groups="account.group_account_manager"
|
||||
class="oe_stat_button"
|
||||
icon="fa-link"
|
||||
attrs="{'invisible':['|',('unreconciled', '=', False), ('state', '!=', 'done')]}"
|
||||
>
|
||||
</button>
|
||||
</div>
|
||||
</field>
|
||||
</record>
|
||||
<record id="purchase_order_view_search" model="ir.ui.view">
|
||||
<field name="name">Custom Purchase Unreconciled Search</field>
|
||||
<field name="model">purchase.order</field>
|
||||
<field name="inherit_id" ref="purchase.purchase_order_view_search" />
|
||||
<field name="arch" type="xml">
|
||||
<filter name="order_date" position="after">
|
||||
<filter
|
||||
name="unreconciled"
|
||||
string="Unreconciled"
|
||||
domain="[('unreconciled','=', True)]"
|
||||
groups="account.group_account_manager"
|
||||
help="Purchase orders with unreconciled journal items."
|
||||
/>
|
||||
<filter
|
||||
name="accounting_review"
|
||||
string="To Review By Accounting"
|
||||
domain="[('unreconciled','=', True), ('invoice_status', '=', 'invoiced'), ('is_shipped', '=', True)]"
|
||||
groups="account.group_account_manager"
|
||||
help="No unreconciled journal items should remain after receive and bill"
|
||||
/>
|
||||
</filter>
|
||||
</field>
|
||||
</record>
|
||||
</odoo>
|
||||
40
purchase_unreconciled/views/res_config_settings_view.xml
Normal file
40
purchase_unreconciled/views/res_config_settings_view.xml
Normal file
@@ -0,0 +1,40 @@
|
||||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<odoo>
|
||||
<record id="res_config_settings_view_form" model="ir.ui.view">
|
||||
<field name="name">res.config.settings.view.form.purchase.unreconciled</field>
|
||||
<field name="model">res.config.settings</field>
|
||||
<field name="inherit_id" ref="account.res_config_settings_view_form" />
|
||||
<field name="arch" type="xml">
|
||||
<xpath expr="//div[@id='bank_cash']" position="after">
|
||||
<h2>Purchase Reconciling</h2>
|
||||
<div
|
||||
class="row mt16 o_settings_container"
|
||||
id="account_purchase_unreconciled"
|
||||
>
|
||||
<div
|
||||
class="col-xs-12 col-md-6 o_setting_box"
|
||||
id="account_check_deposit_offsetting_account"
|
||||
>
|
||||
<div class="o_setting_left_pane" />
|
||||
<div class="o_setting_right_pane">
|
||||
<label for="purchase_reconcile_account_id" />
|
||||
<div class="content-group">
|
||||
<field
|
||||
name="purchase_reconcile_account_id"
|
||||
class="o_light_label mt16"
|
||||
/>
|
||||
</div>
|
||||
<label for="purchase_reconcile_journal_id" />
|
||||
<div class="content-group">
|
||||
<field
|
||||
name="purchase_reconcile_journal_id"
|
||||
class="o_light_label mt16"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</xpath>
|
||||
</field>
|
||||
</record>
|
||||
</odoo>
|
||||
Reference in New Issue
Block a user