Tower: upload laundry_management 19.0.19.0.4 (via marketplace)

This commit is contained in:
2026-05-01 15:00:53 +00:00
parent 256c5679c7
commit 5ee5d4f5cd

View File

@@ -0,0 +1,158 @@
"""POS ↔ laundry.order link healthcheck.
Run via:
docker exec -i odoo-laundry-system-odoo19-1 odoo shell \\
-c /etc/odoo/odoo.conf -d dev --no-http \\
< extra-addons/laundry_management/tools/laundry_pos_link_healthcheck.py
Read-only by default. To heal missing links + repair the tracking-code
sequence, set HEAL=True at the top.
What it does
────────────
1. Reports the laundry tracking-code sequence vs MAX(tracking_code) —
this is the most common cause of the silent "no laundry.order
created" symptom (UniqueViolation swallowed by the savepoint).
2. Lists the latest 10 POS orders with linkage status + line scan.
3. Lists the latest 10 laundry.order rows.
4. Counts unlinked POS orders that SHOULD have a laundry.order
(have ≥1 laundry-service line, no settlement-only).
5. Verifies that the laundry-side hand-off is callable as a Cashier-
group user (permission smoke).
6. Optional heal: backfills missing links + repairs the sequence.
"""
import traceback
env = self.env # noqa: F821 — provided by odoo shell
# Toggle to True to backfill missing links + repair the sequence.
HEAL = False
print("=" * 70)
print("POS ↔ LAUNDRY LINK HEALTHCHECK HEAL =", HEAL)
print("=" * 70)
# ── 1. Sequence sanity check ──────────────────────────────────────────
SEQ_CODE = "laundry.order.line.tracking"
seq = env["ir.sequence"].sudo().search([("code", "=", SEQ_CODE)], limit=1)
if seq:
env.cr.execute(
"""SELECT COALESCE(MAX(
CAST(NULLIF(REGEXP_REPLACE(tracking_code,'[^0-9]','','g'),'')
AS INTEGER)
), 0) FROM laundry_order_line WHERE tracking_code IS NOT NULL"""
)
max_existing = env.cr.fetchone()[0] or 0
drift = seq.number_next - (max_existing + 1)
status = "OK" if drift >= 0 else f"BEHIND by {-drift} (collisions imminent)"
print(f"\n[1] sequence {SEQ_CODE}")
print(f" number_next = {seq.number_next} "
f"max_existing = {max_existing} drift = {drift} {status}")
else:
print(f"\n[1] !!! sequence {SEQ_CODE} not found")
# ── 2. Latest 10 POS orders with linkage ─────────────────────────────
print("\n[2] last 10 pos.order with linkage")
env.cr.execute("""
SELECT id, name, pos_reference, partner_id, amount_total, state,
create_date, laundry_order_id
FROM pos_order
ORDER BY id DESC
LIMIT 10
""")
for row in env.cr.dictfetchall():
link = row["laundry_order_id"] or "-"
print(f" pos.id={row['id']:>3} name={(row['name'] or '')[:24]:<24} "
f"ref={(row['pos_reference'] or '-'):<22} "
f"link={link:>4} state={row['state']}")
# ── 3. Latest 10 laundry.order ────────────────────────────────────────
print("\n[3] last 10 laundry.order")
env.cr.execute("""
SELECT id, name, pos_order_id, partner_id, amount_total, amount_due, state
FROM laundry_order
ORDER BY id DESC
LIMIT 10
""")
for row in env.cr.dictfetchall():
print(f" lo.id={row['id']:>3} name={row['name']:<22} "
f"pos={row['pos_order_id'] or '-':>4} "
f"total={row['amount_total']:>7} due={row['amount_due']:>5} "
f"state={row['state']}")
# ── 4. Unlinked POS orders that SHOULD have a laundry.order ──────────
print("\n[4] unlinked POS orders that have laundry-service lines")
unlinked = env["pos.order"].search(
[("laundry_order_id", "=", False)], order="id desc",
)
should_have = []
no_laundry_lines = []
for o in unlinked:
has_laundry = any(
line.product_id
and line.product_id.product_tmpl_id.is_laundry_service
and not line.product_id.product_tmpl_id.is_laundry_settlement
for line in o.lines
)
if has_laundry:
should_have.append(o)
else:
no_laundry_lines.append(o)
print(f" should-have-link (skipped): {len(should_have)}")
print(f" no-laundry-lines (correct skip): {len(no_laundry_lines)}")
for o in should_have[:10]:
print(f" pos.id={o.id:>3} name={o.name!r} partner={o.partner_id.name!r}")
# ── 5. Permission smoke — Cashier can run the hand-off ───────────────
print("\n[5] permission smoke — call _maybe_create_laundry_order as Cashier")
cashier_group = env.ref(
"laundry_management.group_laundry_cashier", raise_if_not_found=False,
)
# Odoo 19: res.groups.users → res.groups.user_ids
group_users = (
getattr(cashier_group, "user_ids", None)
or getattr(cashier_group, "users", None)
) if cashier_group else None
if group_users:
cashier = group_users[0]
print(f" using existing cashier: {cashier.login}")
if should_have:
target = should_have[0]
try:
with env.cr.savepoint():
target.with_user(cashier).sudo()._maybe_create_laundry_order()
print(f" [OK] handoff callable as Cashier on POS {target.id}")
except Exception:
print(f" !!! permission failure:")
traceback.print_exc()
else:
print(" no cashier user available — skipping (admin-only env)")
# ── 6. Optional heal ─────────────────────────────────────────────────
if HEAL:
print("\n[6] HEALING")
if seq:
env["laundry.order.line"].sudo()._repair_tracking_sequence()
seq_after = env["ir.sequence"].sudo().search(
[("code", "=", SEQ_CODE)], limit=1,
)
print(f" sequence repaired → number_next = {seq_after.number_next}")
healed = 0
for o in should_have:
try:
with env.cr.savepoint():
o.sudo()._maybe_create_laundry_order()
if o.laundry_order_id:
healed += 1
except Exception:
print(f" heal failed on POS {o.id}:")
traceback.print_exc()
print(f" healed {healed} of {len(should_have)} missing links")
env.cr.commit()
print(" committed.")
else:
print("\n[6] heal SKIPPED — set HEAL=True at top of script to backfill")
print("\n" + "=" * 70)
print("HEALTHCHECK DONE")
print("=" * 70)