Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[16.0] [IMP] shopinvader_api_cart: Make cart transaction grouping extendable #1575

Open
wants to merge 1 commit into
base: 16.0
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
63 changes: 42 additions & 21 deletions shopinvader_api_cart/routers/cart.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
# Copyright 2024 Camptocamp (http://www.camptocamp.com).
# @author Simone Orsi <simahawk@gmail.com>
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
from collections import OrderedDict
from collections import defaultdict, namedtuple
from typing import Annotated
from uuid import UUID

Expand Down Expand Up @@ -95,6 +95,30 @@ class ShopinvaderApiCartRouterHelper(models.AbstractModel):
_name = "shopinvader_api_cart.cart_router.helper"
_description = "ShopInvader API Cart Router Helper"

def _get_transaction_key(self, transaction: CartTransaction):
"""
Return a namedtuple of values that identify a transaction and match it
against a cart line.

To override this method create a new namedtuple combining the super fields
and the new ones and return it with the values that identify the transaction.

Example:
```python

def _get_transaction_key(self, transaction: CartTransaction):
key = super()._get_transaction_key(transaction)
return namedtuple(
key.__class__.__name__, key._fields + ('my_field',)
)(
*key,
my_field=transaction.my_field,
)
```

"""
return namedtuple("TransactionKey", ["product_id"])(transaction.product_id)

@api.model
def _check_transactions(self, transactions: list[CartTransaction]):
"""Check if the transactions info are valid.
Expand All @@ -109,21 +133,15 @@ def _check_transactions(self, transactions: list[CartTransaction]):
)

@api.model
def _group_transactions_by_product_id(self, transactions: list[CartTransaction]):
def _group_transactions(self, transactions: list[CartTransaction]):
"""
Gather together transactions that are linked to the same product.
Gather together transactions that are linked to the same transaction key.
"""
# take an ordered dict to ensure to create lines into the same
# order as the transactions list
transactions_by_product_id = OrderedDict()
for trans in transactions:
product_id = trans.product_id
transactions = transactions_by_product_id.get(product_id)
if not transactions:
transactions = []
transactions_by_product_id[product_id] = transactions
transactions.append(trans)
return transactions_by_product_id
grouped_transactions = defaultdict(list)
for transaction in transactions:
key = self._get_transaction_key(transaction)
grouped_transactions[key].append(transaction)
return grouped_transactions

@api.model
def _apply_transactions_on_existing_cart_line(
Expand Down Expand Up @@ -231,23 +249,26 @@ def _apply_transactions(self, cart, transactions: list[CartTransaction]):
return
cart.ensure_one()
self._check_transactions(transactions=transactions)
transactions_by_product_id = self._group_transactions_by_product_id(
transactions=transactions
)
update_cmds = []
# prefetch all products
self.env["product.product"].browse(transactions_by_product_id.keys())
self.env["product.product"].browse({tx.product_id for tx in transactions})
grouped_transactions = self._group_transactions(transactions=transactions)
update_cmds = []
# here we avoid that each on change on a line trigger all the
# recompute methods on the SO. These methods will be triggered
# by the orm into the 'write' process
for product_id, trxs in transactions_by_product_id.items():
line = cart._get_cart_line(product_id)
for key, trxs in grouped_transactions.items():
line = cart._get_cart_line(
**self._apply_transactions_creating_new_cart_line_prepare_vals(
cart, trxs, {"product_id": key.product_id}
)
)
if line:
cmd = self._apply_transactions_on_existing_cart_line(line, trxs)
else:
cmd = self._apply_transactions_creating_new_cart_line(cart, trxs)
if cmd:
update_cmds.append(cmd)

all_transaction_uuids = transaction_uuids = [
str(t.uuid) for t in transactions if t.uuid
]
Expand Down
18 changes: 6 additions & 12 deletions shopinvader_sale_cart/models/sale_order.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@


class SaleOrder(models.Model):

_inherit = "sale.order"

uuid = fields.Char(string="EShop Unique identifier", readonly=True)
Expand Down Expand Up @@ -75,31 +74,26 @@ def _create_empty_cart(self, partner_id):
vals = self._prepare_cart(partner_id)
return self.create(vals)

def _get_cart_line(self, product_id):
def _get_cart_line(self, **kwargs):
"""
Return the sale order line of the cart associated to the given product.
Return the sale order line of the cart associated to the given fields.
"""
self.ensure_one()
return self.order_line.filtered(
lambda l, product_id=product_id: l.product_id.id == product_id
)[:1]
return self.order_line.filtered(lambda sol: sol._match_cart_line(**kwargs))[:1]

def _update_cart_lines_from_cart(self, cart):
self.ensure_one()
update_cmds = []
for cart_line in cart.order_line:
line = self._get_cart_line(cart_line.product_id.id)
line = self._get_cart_line(**cart_line.read(load=False)[0])
if line:
new_qty = line.product_uom_qty + cart_line.product_uom_qty
vals = {"product_uom_qty": new_qty}
vals.update(line._play_onchanges_cart_line(vals))
cmd = (1, line.id, vals)
else:
vals = {
"order_id": self.id,
"product_id": cart_line.product_id.id,
"product_uom_qty": cart_line.product_uom_qty,
}
vals = cart_line._prepare_cart_line_transfer_values()
vals["order_id"] = self.id
vals.update(self.env["sale.order.line"]._play_onchanges_cart_line(vals))
cmd = (0, None, vals)
update_cmds.append(cmd)
Expand Down
17 changes: 16 additions & 1 deletion shopinvader_sale_cart/models/sale_order_line.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,24 @@


class SaleOrderLine(models.Model):

_inherit = "sale.order.line"

@api.model
def _play_onchanges_cart_line(self, vals):
return self.sudo().play_onchanges(vals, vals.keys())

def _prepare_cart_line_transfer_values(self):
"""
Prepare the values to create a new cart line from a given cart line,
in case of a cart transfer for example.
"""
return {
"product_id": self.product_id.id,
"product_uom_qty": self.product_uom_qty,
}

def _match_cart_line(self, product_id, **kwargs):
"""
Return True if the given sale order line matches the given fields.
"""
return self.product_id.id == product_id
Loading