Add oca-purchase submodule with 96 purchase modules moved from oca-workflow-process

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
Ernad Husremovic 2025-08-30 18:00:40 +02:00
parent b0628ee8ea
commit 7378b233e9
3994 changed files with 334316 additions and 0 deletions

View file

@ -0,0 +1,46 @@
# Purchase Line Procurement Group
Odoo addon: purchase_line_procurement_group
## Installation
```bash
pip install odoo-bringout-oca-purchase-workflow-purchase_line_procurement_group
```
## Dependencies
This addon depends on:
- purchase_stock
## Manifest Information
- **Name**: Purchase Line Procurement Group
- **Version**: 16.0.1.0.0
- **Category**: Purchase
- **License**: AGPL-3
- **Installable**: False
## Source
Based on [OCA/purchase-workflow](https://github.com/OCA/purchase-workflow) branch 16.0, addon `purchase_line_procurement_group`.
## License
This package maintains the original AGPL-3 license from the upstream Odoo project.
## Documentation
- Overview: doc/OVERVIEW.md
- Architecture: doc/ARCHITECTURE.md
- Models: doc/MODELS.md
- Controllers: doc/CONTROLLERS.md
- Wizards: doc/WIZARDS.md
- Reports: doc/REPORTS.md
- Security: doc/SECURITY.md
- Install: doc/INSTALL.md
- Usage: doc/USAGE.md
- Configuration: doc/CONFIGURATION.md
- Dependencies: doc/DEPENDENCIES.md
- Troubleshooting: doc/TROUBLESHOOTING.md
- FAQ: doc/FAQ.md

View file

@ -0,0 +1,32 @@
# Architecture
```mermaid
flowchart TD
U[Users] -->|HTTP| V[Views and QWeb Templates]
V --> C[Controllers]
V --> W[Wizards Transient Models]
C --> M[Models and ORM]
W --> M
M --> R[Reports]
DX[Data XML] --> M
S[Security ACLs and Groups] -. enforces .-> M
subgraph Purchase_line_procurement_group Module - purchase_line_procurement_group
direction LR
M:::layer
W:::layer
C:::layer
V:::layer
R:::layer
S:::layer
DX:::layer
end
classDef layer fill:#eef8ff,stroke:#6ea8fe,stroke-width:1px
```
Notes
- Views include tree/form/kanban templates and report templates.
- Controllers provide website/portal routes when present.
- Wizards are UI flows implemented with `models.TransientModel`.
- Data XML loads data/demo records; Security defines groups and access.

View file

@ -0,0 +1,3 @@
# Configuration
Refer to Odoo settings for purchase_line_procurement_group. Configure related models, access rights, and options as needed.

View file

@ -0,0 +1,3 @@
# Controllers
This module does not define custom HTTP controllers.

View file

@ -0,0 +1,5 @@
# Dependencies
This addon depends on:
- [purchase_stock](https://github.com/bringout/oca-ocb-warehouse/tree/0ee5ffef60413a71dceb350918ad3fb572ec1875/odoo-bringout-oca-ocb-purchase_stock)

View file

@ -0,0 +1,4 @@
# FAQ
- Q: Which Odoo version? A: 16.0 (OCA/OCB packaged).
- Q: How to enable? A: Start server with --addon purchase_line_procurement_group or install in UI.

View file

@ -0,0 +1,7 @@
# Install
```bash
pip install odoo-bringout-oca-purchase-workflow-purchase_line_procurement_group"
# or
uv pip install odoo-bringout-oca-purchase-workflow-purchase_line_procurement_group"
```

View file

@ -0,0 +1,13 @@
# Models
Detected core models and extensions in purchase_line_procurement_group.
```mermaid
classDiagram
class purchase_order_line
class stock_move
```
Notes
- Classes show model technical names; fields omitted for brevity.
- Items listed under _inherit are extensions of existing models.

View file

@ -0,0 +1,6 @@
# Overview
Packaged Odoo addon: purchase_line_procurement_group. Provides features documented in upstream Odoo 16 under this addon.
- Source: OCA/OCB 16.0, addon purchase_line_procurement_group
- License: LGPL-3

View file

@ -0,0 +1,3 @@
# Reports
This module does not define custom reports.

View file

@ -0,0 +1,8 @@
# Security
This module does not define custom security rules or access controls beyond Odoo defaults.
Default Odoo security applies:
- Base user access through standard groups
- Model access inherited from dependencies
- No custom row-level security rules

View file

@ -0,0 +1,5 @@
# Troubleshooting
- Ensure Python and Odoo environment matches repo guidance.
- Check database connectivity and logs if startup fails.
- Validate that dependent addons listed in DEPENDENCIES.md are installed.

View file

@ -0,0 +1,7 @@
# Usage
Start Odoo including this addon (from repo root):
```bash
python3 scripts/nix_odoo_web_server.py --db-name mydb --addon purchase_line_procurement_group
```

View file

@ -0,0 +1,3 @@
# Wizards
This module does not include UI wizards.

View file

@ -0,0 +1,91 @@
===============================
Purchase Line Procurement Group
===============================
..
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! This file is generated by oca-gen-addon-readme !!
!! changes will be overwritten. !!
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! source digest: sha256:bdf5b1a2feeb4d286b1a4574b6f253c1951af7b1ddd8169b5ee95fd3de8bbf3c
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
.. |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%2Fpurchase--workflow-lightgray.png?logo=github
:target: https://github.com/OCA/purchase-workflow/tree/16.0/purchase_line_procurement_group
:alt: OCA/purchase-workflow
.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png
:target: https://translation.odoo-community.org/projects/purchase-workflow-16-0/purchase-workflow-16-0-purchase_line_procurement_group
: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/purchase-workflow&target_branch=16.0
:alt: Try me on Runboat
|badge1| |badge2| |badge3| |badge4| |badge5|
This module changes the way procurements generate purchase order lines.
If some procurements are run for the same products and locations, but have
different procurement groups, these won't be merged in the same purchase order
line and will instead generate a purchase order line per procurement group.
Moreover this module ensures that generated stock move won't be merged together
if they come from purchase order lines with different procurement groups.
**Table of contents**
.. contents::
:local:
Bug Tracker
===========
Bugs are tracked on `GitHub Issues <https://github.com/OCA/purchase-workflow/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 <https://github.com/OCA/purchase-workflow/issues/new?body=module:%20purchase_line_procurement_group%0Aversion:%2016.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
~~~~~~~
* Camptocamp
Contributors
~~~~~~~~~~~~
* Akim Juillerat <akim.juillerat@camptocamp.com>
* Kitti Upariphutthiphong <kittiu@ecosoft.co.th>
* `Trobz <https://trobz.com>`_:
* Phuc Tran <phuc@trobz.com>
* Jack Le <anlh@trobz.com>
Other credits
~~~~~~~~~~~~~
The migration of this module from 14.0 to 16.0 was financially supported by Camptocamp
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/purchase-workflow <https://github.com/OCA/purchase-workflow/tree/16.0/purchase_line_procurement_group>`_ project on GitHub.
You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.

View file

@ -0,0 +1,13 @@
# Copyright 2018 Camptocamp SA
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
{
"name": "Purchase Line Procurement Group",
"summary": "Group purchase order line according to procurement group",
"version": "16.0.1.0.0",
"category": "Purchase",
"website": "https://github.com/OCA/purchase-workflow",
"author": "Camptocamp, Odoo Community Association (OCA)",
"license": "AGPL-3",
"depends": ["purchase_stock"],
"data": ["views/purchase.xml"],
}

View file

@ -0,0 +1,29 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * purchase_line_procurement_group
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.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: purchase_line_procurement_group
#: model:ir.model.fields,field_description:purchase_line_procurement_group.field_purchase_order_line__procurement_group_id
msgid "Procurement Group"
msgstr "Grupa naručivanja"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_purchase_order_line
msgid "Purchase Order Line"
msgstr "Stavka naloga za nabavu"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_stock_move
msgid "Stock Move"
msgstr "Skladišno kretanje"

View file

@ -0,0 +1,32 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * purchase_line_procurement_group
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2024-02-11 19:34+0000\n"
"Last-Translator: Ivorra78 <informatica@totmaterial.es>\n"
"Language-Team: none\n"
"Language: es\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: nplurals=2; plural=n != 1;\n"
"X-Generator: Weblate 4.17\n"
#. module: purchase_line_procurement_group
#: model:ir.model.fields,field_description:purchase_line_procurement_group.field_purchase_order_line__procurement_group_id
msgid "Procurement Group"
msgstr "Grupo de Adquisiciones"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_purchase_order_line
msgid "Purchase Order Line"
msgstr "Línea de Orden de Compra"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_stock_move
msgid "Stock Move"
msgstr "Movimiento de Existencias"

View file

@ -0,0 +1,35 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * purchase_line_procurement_group
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 12.0\n"
"Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2020-10-04 16:24+0000\n"
"Last-Translator: Yann Papouin <y.papouin@dec-industrie.com>\n"
"Language-Team: none\n"
"Language: fr\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: nplurals=2; plural=n > 1;\n"
"X-Generator: Weblate 3.10\n"
#. module: purchase_line_procurement_group
#: model:ir.model.fields,field_description:purchase_line_procurement_group.field_purchase_order_line__procurement_group_id
msgid "Procurement Group"
msgstr "Groupe d'approvisionnement"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_purchase_order_line
msgid "Purchase Order Line"
msgstr "Ligne de commande d'achat"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_stock_move
msgid "Stock Move"
msgstr "Mouvement de stock"
#~ msgid "Stock Rule"
#~ msgstr "Règle de stock"

View file

@ -0,0 +1,32 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * purchase_line_procurement_group
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2023-12-22 14:36+0000\n"
"Last-Translator: mymage <stefano.consolaro@mymage.it>\n"
"Language-Team: none\n"
"Language: it\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: nplurals=2; plural=n != 1;\n"
"X-Generator: Weblate 4.17\n"
#. module: purchase_line_procurement_group
#: model:ir.model.fields,field_description:purchase_line_procurement_group.field_purchase_order_line__procurement_group_id
msgid "Procurement Group"
msgstr "Gruppo di approvvigionamento"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_purchase_order_line
msgid "Purchase Order Line"
msgstr "Riga ordine di acquisto"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_stock_move
msgid "Stock Move"
msgstr "Movimento di magazzino"

View file

@ -0,0 +1,32 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * purchase_line_procurement_group
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.0\n"
"Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2023-12-05 11:34+0000\n"
"Last-Translator: \"Ivy Liu (QRTL)\" <liuhehe@quartile.co>\n"
"Language-Team: none\n"
"Language: ja\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: nplurals=1; plural=0;\n"
"X-Generator: Weblate 4.17\n"
#. module: purchase_line_procurement_group
#: model:ir.model.fields,field_description:purchase_line_procurement_group.field_purchase_order_line__procurement_group_id
msgid "Procurement Group"
msgstr "調達グループ"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_purchase_order_line
msgid "Purchase Order Line"
msgstr "購買明細行"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_stock_move
msgid "Stock Move"
msgstr "在庫移動"

View file

@ -0,0 +1,29 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * purchase_line_procurement_group
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 16.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: purchase_line_procurement_group
#: model:ir.model.fields,field_description:purchase_line_procurement_group.field_purchase_order_line__procurement_group_id
msgid "Procurement Group"
msgstr ""
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_purchase_order_line
msgid "Purchase Order Line"
msgstr ""
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_stock_move
msgid "Stock Move"
msgstr ""

View file

@ -0,0 +1,35 @@
# Translation of Odoo Server.
# This file contains the translation of the following modules:
# * purchase_line_procurement_group
#
msgid ""
msgstr ""
"Project-Id-Version: Odoo Server 12.0\n"
"Report-Msgid-Bugs-To: \n"
"PO-Revision-Date: 2019-09-02 14:40+0000\n"
"Last-Translator: 黎伟杰 <674416404@qq.com>\n"
"Language-Team: none\n"
"Language: zh_CN\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: \n"
"Plural-Forms: nplurals=1; plural=0;\n"
"X-Generator: Weblate 3.8\n"
#. module: purchase_line_procurement_group
#: model:ir.model.fields,field_description:purchase_line_procurement_group.field_purchase_order_line__procurement_group_id
msgid "Procurement Group"
msgstr "补货组"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_purchase_order_line
msgid "Purchase Order Line"
msgstr "采购订单行"
#. module: purchase_line_procurement_group
#: model:ir.model,name:purchase_line_procurement_group.model_stock_move
msgid "Stock Move"
msgstr "库存移动"
#~ msgid "Stock Rule"
#~ msgstr "库存规则"

View file

@ -0,0 +1,2 @@
from . import purchase_order_line
from . import stock_move

View file

@ -0,0 +1,58 @@
# Copyright 2018 Camptocamp SA
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo import fields, models
class PurchaseOrderLine(models.Model):
_inherit = "purchase.order.line"
procurement_group_id = fields.Many2one("procurement.group")
def _find_candidate(
self,
product_id,
product_qty,
product_uom,
location_id,
name,
origin,
company_id,
values,
):
"""Do not merge PO lines if procurement group is different or not set."""
_self = self
if "group_id" in values:
pg_id = values.get("group_id", False)
if pg_id:
pg_id = pg_id.id
_self = self.filtered(lambda l: l.procurement_group_id.id == pg_id)
return super(PurchaseOrderLine, _self)._find_candidate(
product_id=product_id,
product_qty=product_qty,
product_uom=product_uom,
location_id=location_id,
name=name,
origin=origin,
company_id=company_id,
values=values,
)
def _prepare_stock_moves(self, picking):
res = super()._prepare_stock_moves(picking)
if res and res[0] and "group_id" in res[0]:
res[0]["group_id"] = (
self.procurement_group_id.id or self.order_id.group_id.id
)
return res
def _prepare_purchase_order_line_from_procurement(
self, product_id, product_qty, product_uom, company_id, values, po
):
"""Add procurement group to values"""
res = super()._prepare_purchase_order_line_from_procurement(
product_id, product_qty, product_uom, company_id, values, po
)
procurement_group = values.get("group_id")
if procurement_group:
res["procurement_group_id"] = procurement_group.id
return res

View file

@ -0,0 +1,14 @@
# Copyright 2018 Camptocamp SA
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo import api, models
class StockMove(models.Model):
_inherit = "stock.move"
@api.model
def _prepare_merge_moves_distinct_fields(self):
res = super()._prepare_merge_moves_distinct_fields()
res.append("group_id")
return res

View file

@ -0,0 +1,5 @@
* Akim Juillerat <akim.juillerat@camptocamp.com>
* Kitti Upariphutthiphong <kittiu@ecosoft.co.th>
* `Trobz <https://trobz.com>`_:
* Phuc Tran <phuc@trobz.com>
* Jack Le <anlh@trobz.com>

View file

@ -0,0 +1 @@
The migration of this module from 14.0 to 16.0 was financially supported by Camptocamp

View file

@ -0,0 +1,7 @@
This module changes the way procurements generate purchase order lines.
If some procurements are run for the same products and locations, but have
different procurement groups, these won't be merged in the same purchase order
line and will instead generate a purchase order line per procurement group.
Moreover this module ensures that generated stock move won't be merged together
if they come from purchase order lines with different procurement groups.

View file

@ -0,0 +1,441 @@
<?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: https://docutils.sourceforge.io/" />
<title>Purchase Line Procurement Group</title>
<style type="text/css">
/*
:Author: David Goodger (goodger@python.org)
:Id: $Id: html4css1.css 8954 2022-01-20 10:10:25Z milde $
:Copyright: This stylesheet has been placed in the public domain.
Default cascading style sheet for the HTML output of Docutils.
See https://docutils.sourceforge.io/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-line-procurement-group">
<h1 class="title">Purchase Line Procurement Group</h1>
<!-- !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! This file is generated by oca-gen-addon-readme !!
!! changes will be overwritten. !!
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
!! source digest: sha256:bdf5b1a2feeb4d286b1a4574b6f253c1951af7b1ddd8169b5ee95fd3de8bbf3c
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! -->
<p><a class="reference external image-reference" 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 image-reference" 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 image-reference" href="https://github.com/OCA/purchase-workflow/tree/16.0/purchase_line_procurement_group"><img alt="OCA/purchase-workflow" src="https://img.shields.io/badge/github-OCA%2Fpurchase--workflow-lightgray.png?logo=github" /></a> <a class="reference external image-reference" href="https://translation.odoo-community.org/projects/purchase-workflow-16-0/purchase-workflow-16-0-purchase_line_procurement_group"><img alt="Translate me on Weblate" src="https://img.shields.io/badge/weblate-Translate%20me-F47D42.png" /></a> <a class="reference external image-reference" href="https://runboat.odoo-community.org/builds?repo=OCA/purchase-workflow&amp;target_branch=16.0"><img alt="Try me on Runboat" src="https://img.shields.io/badge/runboat-Try%20me-875A7B.png" /></a></p>
<p>This module changes the way procurements generate purchase order lines.
If some procurements are run for the same products and locations, but have
different procurement groups, these wont be merged in the same purchase order
line and will instead generate a purchase order line per procurement group.</p>
<p>Moreover this module ensures that generated stock move wont be merged together
if they come from purchase order lines with different procurement groups.</p>
<p><strong>Table of contents</strong></p>
<div class="contents local topic" id="contents">
<ul class="simple">
<li><a class="reference internal" href="#bug-tracker" id="toc-entry-1">Bug Tracker</a></li>
<li><a class="reference internal" href="#credits" id="toc-entry-2">Credits</a><ul>
<li><a class="reference internal" href="#authors" id="toc-entry-3">Authors</a></li>
<li><a class="reference internal" href="#contributors" id="toc-entry-4">Contributors</a></li>
<li><a class="reference internal" href="#other-credits" id="toc-entry-5">Other credits</a></li>
<li><a class="reference internal" href="#maintainers" id="toc-entry-6">Maintainers</a></li>
</ul>
</li>
</ul>
</div>
<div class="section" id="bug-tracker">
<h1><a class="toc-backref" href="#toc-entry-1">Bug Tracker</a></h1>
<p>Bugs are tracked on <a class="reference external" href="https://github.com/OCA/purchase-workflow/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 to smash it by providing a detailed and welcomed
<a class="reference external" href="https://github.com/OCA/purchase-workflow/issues/new?body=module:%20purchase_line_procurement_group%0Aversion:%2016.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="#toc-entry-2">Credits</a></h1>
<div class="section" id="authors">
<h2><a class="toc-backref" href="#toc-entry-3">Authors</a></h2>
<ul class="simple">
<li>Camptocamp</li>
</ul>
</div>
<div class="section" id="contributors">
<h2><a class="toc-backref" href="#toc-entry-4">Contributors</a></h2>
<ul class="simple">
<li>Akim Juillerat &lt;<a class="reference external" href="mailto:akim.juillerat&#64;camptocamp.com">akim.juillerat&#64;camptocamp.com</a>&gt;</li>
<li>Kitti Upariphutthiphong &lt;<a class="reference external" href="mailto:kittiu&#64;ecosoft.co.th">kittiu&#64;ecosoft.co.th</a>&gt;</li>
<li><dl class="first docutils">
<dt><a class="reference external" href="https://trobz.com">Trobz</a>:</dt>
<dd><ul class="first last">
<li>Phuc Tran &lt;<a class="reference external" href="mailto:phuc&#64;trobz.com">phuc&#64;trobz.com</a>&gt;</li>
<li>Jack Le &lt;<a class="reference external" href="mailto:anlh&#64;trobz.com">anlh&#64;trobz.com</a>&gt;</li>
</ul>
</dd>
</dl>
</li>
</ul>
</div>
<div class="section" id="other-credits">
<h2><a class="toc-backref" href="#toc-entry-5">Other credits</a></h2>
<p>The migration of this module from 14.0 to 16.0 was financially supported by Camptocamp</p>
</div>
<div class="section" id="maintainers">
<h2><a class="toc-backref" href="#toc-entry-6">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/purchase-workflow/tree/16.0/purchase_line_procurement_group">OCA/purchase-workflow</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_po_line_proc_group

View file

@ -0,0 +1,211 @@
# Copyright 2018 Camptocamp SA
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from odoo.tests import TransactionCase
class TestPOLineProcurementGroup(TransactionCase):
@classmethod
def setUpClass(cls):
super().setUpClass()
cls.env = cls.env(context=dict(cls.env.context, tracking_disable=True))
def _create_orderpoint(product, qty_min, qty_max, location):
orderpoint_model = cls.env["stock.warehouse.orderpoint"]
return orderpoint_model.create(
{
"name": "OP/%s" % product.name,
"product_id": product.id,
"product_min_qty": qty_min,
"product_max_qty": qty_max,
"location_id": location.id,
}
)
cls.stock_location = cls.env.ref("stock.stock_location_stock")
# Create supplier
cls.pyromaniacs = cls.env["res.partner"].create(
{"name": "Pyromaniacs Inc", "company_type": "company"}
)
cls.lighter = (
cls.env["product.template"]
.create(
{
"name": "Lighter",
"type": "product",
"purchase_ok": True,
"seller_ids": [
(
0,
0,
{
"partner_id": cls.pyromaniacs.id,
"min_qty": 1,
"price": 1.0,
},
)
],
}
)
.product_variant_ids
)
cls.warehouse = cls.env.ref("stock.warehouse0")
cls.warehouse.write({"reception_steps": "three_steps"})
wh2 = cls.env["stock.warehouse"].create(
{"name": "WH2", "code": "WH2", "partner_id": False}
)
# Create WH > WH2 PG and route
cls.wh_wh2_pg = cls.env["procurement.group"].create(
{"name": "WH > WH2", "move_type": "direct"}
)
wh_wh2_route = cls.env["stock.route"].create(
{
"name": "WH > WH2",
"product_selectable": True,
"rule_ids": [
(
0,
0,
{
"name": "WH>WH2",
"action": "pull",
"location_dest_id": wh2.lot_stock_id.id,
"location_src_id": cls.warehouse.lot_stock_id.id,
"procure_method": "make_to_order",
"picking_type_id": cls.env.ref(
"stock.picking_type_internal"
).id,
"group_propagation_option": "fixed",
"group_id": cls.wh_wh2_pg.id,
"propagate_cancel": True,
},
)
],
}
)
cls.lighter.write({"route_ids": [(4, wh_wh2_route.id)]})
_create_orderpoint(cls.lighter, 15, 30, cls.warehouse.lot_stock_id)
_create_orderpoint(cls.lighter, 10, 20, wh2.lot_stock_id)
# Force parent store computation after creation of WH2 because location
# quantities are computed using parent_left _right in domain
cls.env["stock.location"]._parent_store_compute()
def test_po_line_find_candidate_with_group(self):
"""Ensure that our `_find_candidate` override gives a right match when
comparing the procurement group"""
def create_po_line_in_new_po(product, qty, group=False):
"""Shortcut to create a draft purchase of a `product`. A new
purchase order is created with a single line."""
po = self.env["purchase.order"].create(
{
"partner_id": self.pyromaniacs.id,
"group_id": group and group.id or False,
}
)
po_line = self.env["purchase.order.line"].create(
{
"order_id": po.id,
"product_id": product.id,
"product_qty": qty,
"product_uom": product.uom_id.id,
"price_unit": 1.0,
"procurement_group_id": group and group.id or False,
}
)
return po_line
def find_candidate(po_lines, values):
"""Shortcut to call `_find_candidate` with pre-filled values"""
return po_lines._find_candidate(
product_id=self.lighter,
product_qty=1.0,
product_uom=self.lighter.uom_id.id,
location_id=self.stock_location,
name="",
origin="",
company_id=self.env.company.id,
values=values,
)
# Procurement group that will be used as a filter
my_group = self.env["procurement.group"].create(
{"name": "Group", "move_type": "direct"}
)
# Create a first purchase for our lighter without procurement group
po1_line = create_po_line_in_new_po(self.lighter, 1.0, False)
# Create a second purchase for our lighter with a dedicated
# procurement group
po2_line = create_po_line_in_new_po(self.lighter, 1.0, my_group)
# Create a pool of purchase order lines
po_lines = po1_line + po2_line
# Check match when no procurement group is given
match_lines_1 = find_candidate(
po_lines,
values={
"orderpoint_id": False,
"propagate_cancel": True,
},
)
self.assertEqual(len(match_lines_1), 1)
self.assertEqual(match_lines_1.id, po1_line.id)
# Check match when an empty procurement group is given
match_lines_2 = find_candidate(
po_lines,
values={
"orderpoint_id": False,
"propagate_cancel": True,
"group_id": False,
},
)
self.assertEqual(len(match_lines_2), 1)
self.assertEqual(match_lines_2.id, po1_line.id)
# Check match when a specific procurement group is given
match_lines_3 = find_candidate(
po_lines,
values={
"orderpoint_id": False,
"propagate_cancel": True,
"group_id": my_group,
},
)
self.assertEqual(len(match_lines_3), 1)
self.assertEqual(match_lines_3.id, po2_line.id)
def test_po_line_proc_group(self):
# Ensure PO lines generated by the scheduler have proper PG
self.env["procurement.group"].run_scheduler()
po = self.env["purchase.order"].search(
[("partner_id", "=", self.pyromaniacs.id)]
)
self.assertEqual(len(po.order_line), 2)
for line in po.order_line:
self.assertEqual(line.product_id, self.lighter)
if line.procurement_group_id == self.wh_wh2_pg:
self.assertAlmostEqual(line.product_qty, 20)
else:
self.assertAlmostEqual(line.product_qty, 30)
# Ensure stock moves generated by PO confirmation have proper PG
po.button_confirm()
input_moves = self.env["stock.move"].search(
[
("product_id", "=", self.lighter.id),
("location_dest_id", "=", self.warehouse.wh_input_stock_loc_id.id),
]
)
self.assertEqual(len(input_moves), 2)
for move in input_moves:
self.assertEqual(move.product_id, self.lighter)
if move.group_id == self.wh_wh2_pg:
self.assertAlmostEqual(move.product_uom_qty, 20.0)
else:
self.assertAlmostEqual(move.product_uom_qty, 30.0)

View file

@ -0,0 +1,22 @@
<?xml version="1.0" encoding="utf-8" ?>
<odoo>
<record id="purchase_order_form" model="ir.ui.view">
<field name="name">purchase.order.form.inherit</field>
<field name="model">purchase.order</field>
<field name="inherit_id" ref="purchase.purchase_order_form" />
<field name="arch" type="xml">
<xpath
expr="//notebook//field[@name='order_line']/tree/field[@name='date_planned']"
position="after"
>
<field name="procurement_group_id" group_id="base.group_no_one" />
</xpath>
<xpath
expr="//notebook//field[@name='order_line']/form//field[@name='date_planned']"
position="after"
>
<field name="procurement_group_id" group_id="base.group_no_one" />
</xpath>
</field>
</record>
</odoo>

View file

@ -0,0 +1,42 @@
[project]
name = "odoo-bringout-oca-purchase-workflow-purchase_line_procurement_group"
version = "16.0.0"
description = "Purchase Line Procurement Group - Group purchase order line according to procurement group"
authors = [
{ name = "Ernad Husremovic", email = "hernad@bring.out.ba" }
]
dependencies = [
"odoo-bringout-oca-purchase-workflow-purchase_stock>=16.0.0",
"requests>=2.25.1"
]
readme = "README.md"
requires-python = ">= 3.11"
classifiers = [
"Development Status :: 5 - Production/Stable",
"Intended Audience :: Developers",
"License :: OSI Approved :: GNU Lesser General Public License v3 (LGPLv3)",
"Programming Language :: Python :: 3",
"Programming Language :: Python :: 3.11",
"Programming Language :: Python :: 3.12",
"Topic :: Office/Business",
]
[project.urls]
homepage = "https://github.com/bringout/0"
repository = "https://github.com/bringout/0"
[build-system]
requires = ["hatchling"]
build-backend = "hatchling.build"
[tool.hatch.metadata]
allow-direct-references = true
[tool.hatch.build.targets.wheel]
packages = ["purchase_line_procurement_group"]
[tool.rye]
managed = true
dev-dependencies = [
"pytest>=8.4.1",
]