feat(iot): repackaged Odoo iot modules + Fusion Plating sensor wrapper
Phase A of the IoT initiative — gets the server-side infrastructure
in place before the Raspberry Pi hardware arrives, so the iot admin
UI + /fp/iot/ingest endpoint are ready to accept the first real
temperature reading as soon as the Pi is wired up.
New top-level folder: fusion_iot/
1. **iot_base/** — Odoo S.A. iot_base module, copied from
RePackaged-Odoo verbatim. LGPL-3 upstream, no changes needed.
2. **iot/** — Odoo S.A. iot module, repackaged:
- `models/update.py` neutralised (removed the publisher_warranty
IoT-Box-counting report that phones home to odoo.com for
enterprise licence enforcement)
- `iot_handlers/lib/load_worldline_library.sh` deleted (proprietary
Worldline payment lib fetch from download.odoo.com, not needed)
- `wizard/add_iot_box.py._connect_iot_box_with_pairing_code` —
upstream called odoo.com's iot-proxy to resolve pairing codes;
replaced with a no-op. Pi-side iot_drivers proxy registers
directly with this Odoo server instead.
- Manifest rebranded with an explicit changelog preamble.
3. **fusion_plating_iot/** — new plating-specific wrapper:
- `fp.tank.sensor` — maps an iot.device (or a direct-HTTP-ingest
sensor) to a fusion.plating.tank + fusion.plating.bath.parameter.
Supports DS18B20, PT100/1000, pH, conductivity, level. Per-sensor
alert_min/max overrides.
- `fp.tank.reading` — append-only time-series. On create, evaluates
against sensor's alert range. On in-spec → out-of-spec TRANSITION,
auto-raises a fusion.plating.quality.hold (once per excursion,
no spam during sustained out-of-spec).
- `POST /fp/iot/ingest` — shared-secret HTTP endpoint for sensors
bypassing the Pi proxy. Token via X-FP-IOT-Token header OR body.
Accepts single-reading or batch payloads.
- Menu under Plating → Operations → Sensors & Readings.
- Tank form inherits get a Sensors tab inline.
Deployed to entech. Verified end-to-end:
- Install: iot_base + iot + fusion_plating_iot all 'installed'
- Smoke test: in-spec → out-of-spec → hold raised (HOLD-0010);
continued excursion → NO duplicate hold; back-in-spec → NEW
excursion → NEW hold (HOLD-0011) ✓
- HTTP endpoint: correct token → 200 accepted; wrong token → 401;
unknown device_serial → 404; batch payload → 200 accepted=N ✓
Phase B (when Raspberry Pi hardware arrives): DS18B20 iot_handler
driver for the Pi-side iot_drivers proxy + systemd service on
vanilla Raspberry Pi OS + first live reading from physical probe.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
29
fusion_iot/fusion_plating_iot/views/fp_iot_menu.xml
Normal file
29
fusion_iot/fusion_plating_iot/views/fp_iot_menu.xml
Normal file
@@ -0,0 +1,29 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!--
|
||||
Copyright 2026 Nexa Systems Inc.
|
||||
License OPL-1
|
||||
|
||||
Surface IoT sensors + readings under the existing Plating >
|
||||
Operations menu. Not a top-level app — sensors are an extension
|
||||
of bath/tank management, not a separate concern.
|
||||
-->
|
||||
<odoo>
|
||||
|
||||
<menuitem id="menu_fp_iot_root"
|
||||
name="Sensors & Readings"
|
||||
parent="fusion_plating.menu_fp_operations"
|
||||
sequence="55"/>
|
||||
|
||||
<menuitem id="menu_fp_tank_sensor"
|
||||
name="Tank Sensors"
|
||||
parent="menu_fp_iot_root"
|
||||
action="action_fp_tank_sensor"
|
||||
sequence="10"/>
|
||||
|
||||
<menuitem id="menu_fp_tank_reading"
|
||||
name="Sensor Readings"
|
||||
parent="menu_fp_iot_root"
|
||||
action="action_fp_tank_reading"
|
||||
sequence="20"/>
|
||||
|
||||
</odoo>
|
||||
@@ -0,0 +1,97 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!--
|
||||
Copyright 2026 Nexa Systems Inc.
|
||||
License OPL-1 (Odoo Proprietary License v1.0)
|
||||
-->
|
||||
<odoo>
|
||||
|
||||
<record id="fp_tank_reading_list" model="ir.ui.view">
|
||||
<field name="name">fp.tank.reading.list</field>
|
||||
<field name="model">fp.tank.reading</field>
|
||||
<field name="arch" type="xml">
|
||||
<list string="Sensor Readings"
|
||||
decoration-danger="not in_spec" default_order="reading_at desc">
|
||||
<field name="reading_at"/>
|
||||
<field name="sensor_id"/>
|
||||
<field name="tank_id" optional="show"/>
|
||||
<field name="parameter_id" optional="hide"/>
|
||||
<field name="value"/>
|
||||
<field name="unit"/>
|
||||
<field name="in_spec" widget="boolean_toggle"/>
|
||||
<field name="source" optional="hide"/>
|
||||
<field name="hold_id" optional="show"/>
|
||||
</list>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="fp_tank_reading_form" model="ir.ui.view">
|
||||
<field name="name">fp.tank.reading.form</field>
|
||||
<field name="model">fp.tank.reading</field>
|
||||
<field name="arch" type="xml">
|
||||
<form string="Sensor Reading" create="false">
|
||||
<sheet>
|
||||
<group>
|
||||
<group>
|
||||
<field name="sensor_id"/>
|
||||
<field name="tank_id" readonly="1"/>
|
||||
<field name="parameter_id" readonly="1"/>
|
||||
<field name="source" readonly="1"/>
|
||||
</group>
|
||||
<group>
|
||||
<field name="reading_at"/>
|
||||
<field name="value"/>
|
||||
<field name="unit" readonly="1"/>
|
||||
<field name="in_spec" readonly="1"/>
|
||||
<field name="hold_id" readonly="1"/>
|
||||
</group>
|
||||
</group>
|
||||
</sheet>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="fp_tank_reading_graph" model="ir.ui.view">
|
||||
<field name="name">fp.tank.reading.graph</field>
|
||||
<field name="model">fp.tank.reading</field>
|
||||
<field name="arch" type="xml">
|
||||
<graph string="Readings Trend" type="line">
|
||||
<field name="reading_at" interval="hour"/>
|
||||
<field name="value" type="measure"/>
|
||||
</graph>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="fp_tank_reading_search" model="ir.ui.view">
|
||||
<field name="name">fp.tank.reading.search</field>
|
||||
<field name="model">fp.tank.reading</field>
|
||||
<field name="arch" type="xml">
|
||||
<search>
|
||||
<field name="sensor_id"/>
|
||||
<field name="tank_id"/>
|
||||
<field name="parameter_id"/>
|
||||
<filter name="out_of_spec" string="Out of Spec"
|
||||
domain="[('in_spec', '=', False)]"/>
|
||||
<filter name="today" string="Today"
|
||||
domain="[('reading_at', '>=', (context_today()).strftime('%Y-%m-%d'))]"/>
|
||||
<filter name="last_24h" string="Last 24h"
|
||||
domain="[('reading_at', '>=', (datetime.datetime.now() - datetime.timedelta(hours=24)).strftime('%Y-%m-%d %H:%M:%S'))]"/>
|
||||
<group>
|
||||
<filter name="by_sensor" string="Sensor"
|
||||
context="{'group_by': 'sensor_id'}"/>
|
||||
<filter name="by_tank" string="Tank"
|
||||
context="{'group_by': 'tank_id'}"/>
|
||||
<filter name="by_day" string="Day"
|
||||
context="{'group_by': 'reading_at:day'}"/>
|
||||
</group>
|
||||
</search>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="action_fp_tank_reading" model="ir.actions.act_window">
|
||||
<field name="name">Sensor Readings</field>
|
||||
<field name="res_model">fp.tank.reading</field>
|
||||
<field name="view_mode">list,graph,form</field>
|
||||
<field name="search_view_id" ref="fp_tank_reading_search"/>
|
||||
</record>
|
||||
|
||||
</odoo>
|
||||
126
fusion_iot/fusion_plating_iot/views/fp_tank_sensor_views.xml
Normal file
126
fusion_iot/fusion_plating_iot/views/fp_tank_sensor_views.xml
Normal file
@@ -0,0 +1,126 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!--
|
||||
Copyright 2026 Nexa Systems Inc.
|
||||
License OPL-1 (Odoo Proprietary License v1.0)
|
||||
-->
|
||||
<odoo>
|
||||
|
||||
<!-- ===== List ===== -->
|
||||
<record id="fp_tank_sensor_list" model="ir.ui.view">
|
||||
<field name="name">fp.tank.sensor.list</field>
|
||||
<field name="model">fp.tank.sensor</field>
|
||||
<field name="arch" type="xml">
|
||||
<list string="Tank Sensors" decoration-danger="not last_reading_in_spec and last_reading_at"
|
||||
decoration-muted="not active">
|
||||
<field name="name"/>
|
||||
<field name="device_kind"/>
|
||||
<field name="tank_id"/>
|
||||
<field name="bath_id" optional="show"/>
|
||||
<field name="parameter_id"/>
|
||||
<field name="device_serial" optional="show"/>
|
||||
<field name="iot_device_id" optional="hide"/>
|
||||
<field name="last_reading_value"/>
|
||||
<field name="last_reading_at"/>
|
||||
<field name="last_reading_in_spec" widget="boolean_toggle"/>
|
||||
<field name="reading_count"/>
|
||||
<field name="active" column_invisible="1"/>
|
||||
</list>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<!-- ===== Form ===== -->
|
||||
<record id="fp_tank_sensor_form" model="ir.ui.view">
|
||||
<field name="name">fp.tank.sensor.form</field>
|
||||
<field name="model">fp.tank.sensor</field>
|
||||
<field name="arch" type="xml">
|
||||
<form string="Tank Sensor">
|
||||
<header/>
|
||||
<sheet>
|
||||
<div class="oe_button_box" name="button_box">
|
||||
<button name="action_view_readings" type="object"
|
||||
class="oe_stat_button" icon="fa-line-chart">
|
||||
<field name="reading_count" widget="statinfo"
|
||||
string="Readings"/>
|
||||
</button>
|
||||
</div>
|
||||
<widget name="web_ribbon" title="In Spec"
|
||||
invisible="not last_reading_in_spec or not last_reading_at"
|
||||
bg_color="text-bg-success"/>
|
||||
<widget name="web_ribbon" title="OUT OF SPEC"
|
||||
invisible="last_reading_in_spec or not last_reading_at"
|
||||
bg_color="text-bg-danger"/>
|
||||
<div class="oe_title">
|
||||
<h1><field name="name" placeholder="e.g. Tank 3 — ENP temp"/></h1>
|
||||
</div>
|
||||
<group>
|
||||
<group string="Hardware">
|
||||
<field name="device_kind"/>
|
||||
<field name="device_serial" placeholder="28-abc123def456"/>
|
||||
<field name="iot_device_id"
|
||||
options="{'no_create': True}"
|
||||
help="Optional — the iot.device auto-registered by the Pi proxy."/>
|
||||
<field name="active"/>
|
||||
</group>
|
||||
<group string="Location">
|
||||
<field name="tank_id" options="{'no_create': True}"/>
|
||||
<field name="bath_id" options="{'no_create': True}"/>
|
||||
<field name="parameter_id" options="{'no_create': True}"/>
|
||||
</group>
|
||||
</group>
|
||||
<group string="Alerting">
|
||||
<group>
|
||||
<field name="alert_on_out_of_spec"/>
|
||||
<field name="alert_min_override"
|
||||
help="Leave 0 to inherit from the bath parameter's target_min."/>
|
||||
<field name="alert_max_override"
|
||||
help="Leave 0 to inherit from the bath parameter's target_max."/>
|
||||
</group>
|
||||
<group string="Most Recent Reading">
|
||||
<field name="last_reading_value" readonly="1"/>
|
||||
<field name="last_reading_at" readonly="1"/>
|
||||
<field name="last_reading_in_spec" readonly="1" widget="boolean_toggle"/>
|
||||
</group>
|
||||
</group>
|
||||
</sheet>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<!-- ===== Search ===== -->
|
||||
<record id="fp_tank_sensor_search" model="ir.ui.view">
|
||||
<field name="name">fp.tank.sensor.search</field>
|
||||
<field name="model">fp.tank.sensor</field>
|
||||
<field name="arch" type="xml">
|
||||
<search>
|
||||
<field name="name"/>
|
||||
<field name="device_serial"/>
|
||||
<field name="tank_id"/>
|
||||
<field name="parameter_id"/>
|
||||
<filter name="out_of_spec" string="Out of Spec"
|
||||
domain="[('last_reading_in_spec', '=', False),
|
||||
('last_reading_at', '!=', False)]"/>
|
||||
<filter name="alerting_on" string="Alerting Enabled"
|
||||
domain="[('alert_on_out_of_spec', '=', True)]"/>
|
||||
<filter name="inactive" string="Archived"
|
||||
domain="[('active', '=', False)]"/>
|
||||
<group>
|
||||
<filter name="by_tank" string="Tank"
|
||||
context="{'group_by': 'tank_id'}"/>
|
||||
<filter name="by_parameter" string="Parameter"
|
||||
context="{'group_by': 'parameter_id'}"/>
|
||||
<filter name="by_kind" string="Sensor Type"
|
||||
context="{'group_by': 'device_kind'}"/>
|
||||
</group>
|
||||
</search>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<!-- ===== Action ===== -->
|
||||
<record id="action_fp_tank_sensor" model="ir.actions.act_window">
|
||||
<field name="name">Tank Sensors</field>
|
||||
<field name="res_model">fp.tank.sensor</field>
|
||||
<field name="view_mode">list,form</field>
|
||||
<field name="search_view_id" ref="fp_tank_sensor_search"/>
|
||||
</record>
|
||||
|
||||
</odoo>
|
||||
@@ -0,0 +1,39 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!--
|
||||
Copyright 2026 Nexa Systems Inc.
|
||||
License OPL-1
|
||||
|
||||
Surface IoT sensors inline on the existing fusion.plating.tank form
|
||||
so the bath operator sees live sensor status in context, not in a
|
||||
separate app.
|
||||
-->
|
||||
<odoo>
|
||||
|
||||
<record id="fusion_plating_tank_form_iot_inherit" model="ir.ui.view">
|
||||
<field name="name">fusion.plating.tank.form.iot</field>
|
||||
<field name="model">fusion.plating.tank</field>
|
||||
<field name="inherit_id" ref="fusion_plating.view_fp_tank_form"/>
|
||||
<field name="arch" type="xml">
|
||||
<xpath expr="//sheet" position="inside">
|
||||
<notebook>
|
||||
<page string="Sensors" name="iot_sensors">
|
||||
<field name="x_fc_sensor_ids" context="{'default_tank_id': id}">
|
||||
<list editable="bottom"
|
||||
decoration-danger="not last_reading_in_spec and last_reading_at">
|
||||
<field name="name"/>
|
||||
<field name="device_kind"/>
|
||||
<field name="device_serial"/>
|
||||
<field name="parameter_id"/>
|
||||
<field name="last_reading_value"/>
|
||||
<field name="last_reading_at" readonly="1"/>
|
||||
<field name="last_reading_in_spec" widget="boolean_toggle"/>
|
||||
<field name="alert_on_out_of_spec" widget="boolean_toggle"/>
|
||||
</list>
|
||||
</field>
|
||||
</page>
|
||||
</notebook>
|
||||
</xpath>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
</odoo>
|
||||
Reference in New Issue
Block a user