Files
Odoo-Modules/fusion_plating/fusion_plating_jobs/controllers/process_tree.py
gsinghpal 71376228cb feat(jobs): Phase 6 lean — scan controller + process-tree JSON endpoint
Phase 6 originally scoped the full operator UI rewrite (Plant
Overview, Tablet, Manager Dashboard, Process Tree). Tailscale SSH
to entech is currently unavailable, so live in-browser
verification of OWL/JS components isn't possible. Shipping a lean
Phase 6 with the data-layer pieces:

1. /fp/job/<id> scan controller — when a user scans a fp.job
   sticker, lands them on the fp.job form (or the process tree
   action once that's wired). Mirrors fusion_plating_reports' /fp/wo/
   pattern.

2. /fp/jobs/process_tree JSON endpoint — returns the recipe tree
   serialized with each node tagged by its fp.job.step state,
   ready for an OWL component to render. The component itself is
   deferred (see README.md).

The bigger UI deferrals (kanban, tablet, manager dashboard) are
documented in README.md. They get their own focused project after
cutover — the data layer is complete, so they can land
incrementally without touching fp.job/fp.job.step.

Tests verify controller imports + serialization shape (no HTTP
because TransactionCase doesn't easily simulate request context).

Manifest 19.0.1.8.0 → 19.0.1.9.0.

Part of: native job model migration (spec 2026-04-25)
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-25 00:08:50 -04:00

49 lines
1.9 KiB
Python

# -*- coding: utf-8 -*-
# Copyright 2026 Nexa Systems Inc.
# License OPL-1 (Odoo Proprietary License v1.0)
#
# /fp/jobs/process_tree — JSON endpoint that returns the recipe tree
# for a given fp.job, with each node tagged by the matching
# fp.job.step (if any) and its current state.
from odoo import http
from odoo.http import request
class FpJobProcessTreeController(http.Controller):
@http.route('/fp/jobs/process_tree', type='jsonrpc', auth='user', website=False)
def fp_jobs_process_tree(self, job_id, **kwargs):
Job = request.env['fp.job']
job = Job.browse(int(job_id)).exists()
if not job:
return {'error': 'Job not found'}
# Map recipe_node_id -> step
step_by_node = {s.recipe_node_id.id: s for s in job.step_ids if s.recipe_node_id}
def serialize(node):
step = step_by_node.get(node.id)
return {
'id': node.id,
'name': node.name,
'node_type': node.node_type,
'sequence': node.sequence,
'step_id': step.id if step else None,
'step_state': step.state if step else None,
'step_assigned_user': step.assigned_user_id.name if step and step.assigned_user_id else None,
'duration_expected': step.duration_expected if step else node.estimated_duration,
'duration_actual': step.duration_actual if step else 0.0,
'children': [serialize(c) for c in node.child_ids.sorted('sequence')],
}
return {
'job_name': job.name,
'partner': job.partner_id.name,
'state': job.state,
'qty': job.qty,
'recipe_name': job.recipe_id.name if job.recipe_id else '',
'progress_pct': job.step_progress_pct,
'tree': serialize(job.recipe_id) if job.recipe_id else None,
}