[ADD] purchase_unreconciled

This commit is contained in:
AaronHForgeFlow
2021-11-23 17:08:24 +01:00
committed by JasminSForgeFlow
parent 2af6e46c54
commit a571f061ee
16 changed files with 1278 additions and 0 deletions

View 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.

View File

@@ -0,0 +1 @@
from . import models

View 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"],
}

View File

@@ -0,0 +1,4 @@
from . import purchase_order
from . import company
from . import res_config_settings
from . import account_move_line

View 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
)

View 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"
)

View 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()

View 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
)

View 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>

View 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.

View 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.

View 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 POs 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. &lt;<a class="reference external" href="mailto:contact&#64;forgeflow.com">contact&#64;forgeflow.com</a>&gt;<ul>
<li>Lois Rilo &lt;<a class="reference external" href="mailto:lois.rilo&#64;forgeflow.com">lois.rilo&#64;forgeflow.com</a>&gt;</li>
<li>Aaron Henriquez &lt;<a class="reference external" href="mailto:ahenriquez&#64;forgeflow.com">ahenriquez&#64;forgeflow.com</a>&gt;</li>
<li>Miquel Raich &lt;<a class="reference external" href="mailto:miquel.raich&#64;forgeflow.com">miquel.raich&#64;forgeflow.com</a>&gt;</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>

View File

@@ -0,0 +1 @@
from . import test_purchase_unreconciled

View 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"))

View 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>

View 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>