Billing Entity Management #66

Merged
rixx merged 32 commits from 54-billing-entity-management into main 2025-06-22 15:42:14 +00:00
Showing only changes of commit 29b630ea2b - Show all commits

View file

@ -91,26 +91,40 @@ CLIENT = OdooClient()
def get_invoice_addresses(user): def get_invoice_addresses(user):
"""Used during organization creation: retrieves all invoice """Used during organization creation: retrieves all invoice
addresses the user owns or is connected to from the Odoo API.""" addresses the user owns or is connected to from the Odoo API."""
or_conditions = [ # Were building our conditions in order:
("email", "ilike", user.email), # - in exceptions, users may be using a billing accounts email
("child_ids.email", "ilike", user.email), # - if the user is associated with an odoo user, return all billing
] # addresses / organizations created by the user
# - if the user is associated with an odoo contact, return all billing
# addresses with the same parent_id
or_conditions = [("email", "ilike", email)]
# Attempt to find the Odoo user ID and add condition for records created by this user email = user if isinstance(user, str) else user.email
try: odoo_users = CLIENT.search_read(
odoo_users = odoo_request( model="res.users",
model="res.users", domain=[("login", "=", email)],
method="search_read", fields=["id"],
domain=[("login", "=", user.email)], limit=1,
fields=["id"], )
limit=1, # Expecting at most one user if odoo_users and (uid := odoo_users[0].get("id")):
) or_conditions.append(("create_uid", "=", uid))
if odoo_users and (uid := odoo_users[0].get("id")):
or_conditions.append(("create_uid", "=", uid)) odoo_contacts = CLIENT.search_read(
except Exception: model="res.partner",
pass domain=[
("company_type", "=", "person"),
("type", "=", "contact"),
("email", "ilike", email),
],
fields=["id", "parent_id"],
)
if odoo_contacts:
for contact in odoo_contacts:
or_conditions.append(("parent_id", "=", contact["parent_id"][0]))
if len(or_conditions) > 1:
or_conditions = ["|"] * (len(or_conditions) - 1) + or_conditions
user_conditions = ["|"] * (len(or_conditions) - 1) + or_conditions
# The domain requires the partner to be an invoice address, that is: # The domain requires the partner to be an invoice address, that is:
# Of the company_type=person, and type=invoice. # Of the company_type=person, and type=invoice.
# If we were searching for an existing organization, we would also have to # If we were searching for an existing organization, we would also have to
@ -118,12 +132,11 @@ def get_invoice_addresses(user):
domain = [ domain = [
("company_type", "=", "person"), ("company_type", "=", "person"),
tobru marked this conversation as resolved Outdated

Enhance the domain to include ('active','=',True) to make sure the user is active in Odoo.

Enhance the domain to include `('active','=',True)` to make sure the user is active in Odoo.
("type", "=", "invoice"), ("type", "=", "invoice"),
] + user_conditions ] + or_conditions
try: try:
invoice_addresses = odoo_request( invoice_addresses = CLIENT.search_read(
tobru marked this conversation as resolved Outdated

A user in the Servala Portal which matches an internal user in Odoo should be able to see all invoice addresses, like they do when they log in to the Odoo backend. I figured out that the "Internal Users" filter in Odoo uses the domain ('share', '=', False) (L342) so let's use the same to decide if the user can see all invoice addresses. There are also "Portal Users" in Odoo, they should not see all invoice addresses and behave like a normal Servala Portal user.

A user in the Servala Portal which matches an internal user in Odoo should be able to see all invoice addresses, like they do when they log in to the Odoo backend. I figured out that the "Internal Users" filter in Odoo uses the domain `('share', '=', False)` ([L342](https://github.com/odoo/odoo/blob/16.0/odoo/addons/base/views/res_users_views.xml#L342)) so let's use the same to decide if the user can see all invoice addresses. There are also "Portal Users" in Odoo, they should not see all invoice addresses and behave like a normal Servala Portal user.
model="res.partner", model="res.partner",
method="search_read",
domain=domain, domain=domain,
fields=ADDRESS_FIELDS, fields=ADDRESS_FIELDS,
) )