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>
49 lines
1.9 KiB
Python
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,
|
|
}
|