Add conflict-of-interest check engine; gate stage advancement at Intake

- New fl.conflict.check model: screens petitioner/respondent/party_ids names
  against parties on other open cases (exact partner match + difflib fuzzy
  match at 0.85 threshold); skips folded/closed stages
- Runs automatically as the first action in fl.case.create; logs conflicts to
  chatter with matched-case detail and never silently passes
- fl.case gains conflict_check_passed/conflict_check_id/conflict_check_ids;
  write() blocks advancing stage_id past Intake until the check passes
- Admin-only action_override requires a written justification, stamps user/date,
  and flips conflict_check_passed True with a chatter audit entry
- Add conflict check form/tree/search views, action, Cases sub-menu item,
  case form banner + Run Conflict Check button, and Kanban conflict badge
- ACL entries for fl.conflict.check (admin full, paralegal no-delete)
- Finish Claude migration cleanup in fl_analysis.py (model_used default,
  docstring/help text)
- Add .gitignore for Python artifacts

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-05-28 18:19:32 +00:00
parent b8ab8494c7
commit 7bc0cc8554
10 changed files with 431 additions and 10 deletions

6
.gitignore vendored Normal file
View File

@@ -0,0 +1,6 @@
__pycache__/
*.py[cod]
*.egg-info/
.idea/
.vscode/
*.swp

View File

@@ -55,6 +55,7 @@
'views/fl_statute_views.xml',
'views/fl_wizard_views.xml',
'views/fl_discovery_suggest_views.xml',
'views/fl_conflict_check_views.xml',
'views/menu_views.xml',
# Phase 4 — QWeb PDF Reports
'report/report_financial_affidavit_short.xml',

View File

@@ -15,3 +15,4 @@ from . import fl_analysis
from . import fl_ai_engine
from . import fl_argument
from . import fl_case
from . import fl_conflict_check

View File

@@ -2,10 +2,7 @@ from odoo import fields, models
class FlAnalysis(models.Model):
"""
Phase 5 — Full Ollama AI analysis implementation.
Phase 1: Stub with fields required by fl_case computed fields.
"""
"""AI case analysis record. Written by the Claude API engine (fl.ai.engine)."""
_name = 'fl.analysis'
_description = 'AI Case Analysis Result'
_order = 'create_date desc'
@@ -19,7 +16,7 @@ class FlAnalysis(models.Model):
)
model_used = fields.Char(
string='AI Model',
default='llama3.1'
default='claude-sonnet-4-20250514'
)
# ── Results (referenced by fl_case related fields) ─────────────────────
@@ -67,7 +64,7 @@ class FlAnalysis(models.Model):
], string='Case Complexity')
raw_response = fields.Text(
string='Raw AI Response',
help='Full JSON response from Ollama — for debugging'
help='Full JSON response from the Claude API — for debugging'
)
error_message = fields.Text(
string='Error (if analysis failed)'

View File

@@ -1,4 +1,5 @@
from odoo import _, api, fields, models
from odoo.exceptions import UserError
from dateutil.relativedelta import relativedelta
@@ -86,6 +87,23 @@ class FlCase(models.Model):
)
active = fields.Boolean(default=True)
# ══════════════════════════════════════════════════════════════════════
# CONFLICT OF INTEREST
# ══════════════════════════════════════════════════════════════════════
conflict_check_passed = fields.Boolean(
string='Conflict Check Passed',
default=False, tracking=True,
help='Case cannot advance past Intake until the conflict-of-interest '
'check passes (or is overridden by an administrator).'
)
conflict_check_id = fields.Many2one(
'fl.conflict.check', string='Latest Conflict Check', readonly=True
)
conflict_check_ids = fields.One2many(
'fl.conflict.check', 'case_id', string='Conflict Checks'
)
# ══════════════════════════════════════════════════════════════════════
# PARTIES
# ══════════════════════════════════════════════════════════════════════
@@ -794,6 +812,8 @@ class FlCase(models.Model):
) or _('New')
records = super().create(vals_list)
for record in records:
# 0. Conflict-of-interest screen — runs before any other action
self.env['fl.conflict.check'].run_check(record)
# 1. Create Odoo project for task management
record._create_case_project()
# 2. Generate deadlines if filing date already set
@@ -807,6 +827,20 @@ class FlCase(models.Model):
return records
def write(self, vals):
# Block advancing past Intake until conflict check has passed
if vals.get('stage_id'):
new_stage = self.env['fl.case.stage'].browse(vals['stage_id'])
intake = self.env.ref(
'activeblue_familylaw.fl_stage_intake', raise_if_not_found=False
)
if intake and new_stage.sequence > intake.sequence:
for rec in self:
if not rec.conflict_check_passed:
raise UserError(_(
"Cannot advance case %s past Intake: the "
"conflict-of-interest check has not passed. Resolve or "
"override the conflict before changing the stage."
) % rec.name)
result = super().write(vals)
# Recalculate service-anchored deadlines when service_date is set/changed
if 'service_date' in vals:
@@ -1090,3 +1124,15 @@ class FlCase(models.Model):
'target': 'new',
'context': {'default_case_id': self.id},
}
def action_run_conflict_check(self):
self.ensure_one()
check = self.env['fl.conflict.check'].run_check(self)
return {
'type': 'ir.actions.act_window',
'name': 'Conflict Check Result',
'res_model': 'fl.conflict.check',
'res_id': check.id,
'view_mode': 'form',
'target': 'new',
}

View File

@@ -0,0 +1,215 @@
import difflib
import logging
from odoo import _, api, fields, models
from odoo.exceptions import UserError
_logger = logging.getLogger(__name__)
# Fuzzy name-match threshold (difflib.SequenceMatcher ratio). Per spec.
FUZZY_THRESHOLD = 0.85
class FlConflictCheck(models.Model):
_name = 'fl.conflict.check'
_description = 'Conflict of Interest Check'
_inherit = ['mail.thread']
_order = 'create_date desc'
_rec_name = 'name'
name = fields.Char(
string='Reference', compute='_compute_name', store=True
)
case_id = fields.Many2one(
'fl.case', string='Case', required=True,
ondelete='cascade', index=True
)
check_date = fields.Datetime(
string='Check Date', default=fields.Datetime.now, readonly=True
)
status = fields.Selection([
('clear', 'Clear — No Conflict'),
('conflict', 'Conflict Detected'),
('override', 'Conflict Overridden'),
], string='Status', default='clear', required=True, tracking=True)
matched_case_ids = fields.Many2many(
'fl.case', 'fl_conflict_check_matched_case_rel',
'conflict_id', 'matched_case_id',
string='Conflicting Cases'
)
match_detail = fields.Text(string='Match Detail', readonly=True)
# ── Override (group_admin only) ────────────────────────────────────────
override_justification = fields.Text(
string='Override Justification',
help='Written justification required to override a detected conflict.'
)
override_user_id = fields.Many2one(
'res.users', string='Overridden By', readonly=True
)
override_date = fields.Datetime(string='Override Date', readonly=True)
@api.depends('case_id', 'case_id.name')
def _compute_name(self):
for rec in self:
rec.name = _('Conflict Check — %s') % (rec.case_id.name or _('New'))
# ──────────────────────────────────────────────────────────────────────
# Engine
# ──────────────────────────────────────────────────────────────────────
@api.model
def run_check(self, case):
"""
Run a conflict-of-interest screen for the given case.
Creates and returns an fl.conflict.check record, sets
case.conflict_check_passed and links the latest check.
Never silently passes a conflict — always surfaces it to chatter.
"""
names = self._collect_party_names(case)
matches = self._find_conflicts(case, names)
check = self.create({
'case_id': case.id,
'status': 'conflict' if matches else 'clear',
'matched_case_ids': (
[(6, 0, list({m['case_id'] for m in matches}))] if matches else False
),
'match_detail': (
self._format_matches(matches) if matches else 'No conflicts found.'
),
})
case.conflict_check_id = check.id
if matches:
case.conflict_check_passed = False
case.message_post(
body=(
'<div style="background:#f8d7da;border:2px solid #dc3545;'
'padding:12px;border-radius:4px;">'
'<h4 style="color:#721c24;margin:0 0 8px 0;">'
'🚩 CONFLICT OF INTEREST DETECTED</h4>'
'<p>One or more parties on this case match parties on other '
'open cases. This case <b>cannot advance past Intake</b> until '
'the conflict is resolved or overridden by an administrator.</p>'
'<pre style="white-space:pre-wrap;margin:0;">'
f'{self._format_matches(matches)}</pre>'
'</div>'
),
subtype_xmlid='mail.mt_comment',
)
else:
case.conflict_check_passed = True
case.message_post(
body='✅ Conflict-of-interest check clear — no matching parties on other open cases.',
subtype_xmlid='mail.mt_note',
)
_logger.info(
"FL Conflict Check: case %s%s (%d match(es))",
case.id, check.status, len(matches)
)
return check
def _collect_party_names(self, case):
"""
Collect (partner_id, name, role_label) tuples for every party on a case.
Pulls from petitioner_id, respondent_id, and party_ids — deduplicated by
partner. Drops entries with no name.
"""
entries = []
seen = set()
def _add(partner, role_label):
if partner and partner.id not in seen and partner.name:
entries.append((partner.id, partner.name, role_label))
seen.add(partner.id)
_add(case.petitioner_id, 'Petitioner')
_add(case.respondent_id, 'Respondent')
for party in case.party_ids:
role_label = dict(party._fields['role'].selection).get(party.role, '')
_add(party.partner_id, role_label)
return entries
def _find_conflicts(self, case, names):
"""
Compare this case's party names against parties on other open cases.
Exact partner-id match OR fuzzy name match (>= FUZZY_THRESHOLD) flags a
conflict. Folded stages (Closed, Referred to Attorney) are treated as
not-open and skipped. Returns a list of match dicts.
"""
if not names:
return []
other_cases = self.env['fl.case'].search([
('id', '!=', case.id),
('active', '=', True),
])
matches = []
for other in other_cases:
if other.stage_id and other.stage_id.fold:
continue
other_names = self._collect_party_names(other)
for pid, pname, prole in names:
for opid, opname, oprole in other_names:
if pid == opid:
ratio = 1.0
else:
ratio = difflib.SequenceMatcher(
None,
(pname or '').lower().strip(),
(opname or '').lower().strip(),
).ratio()
if ratio >= FUZZY_THRESHOLD:
matches.append({
'case_id': other.id,
'case_ref': other.name,
'new_party': f'{pname} ({prole})',
'existing_party': f'{opname} ({oprole})',
'score': round(ratio, 2),
})
return matches
def _format_matches(self, matches):
lines = []
for m in matches:
lines.append(
f"{m['new_party']}{m['existing_party']} "
f"on case {m['case_ref']} (similarity {m['score']:.0%})"
)
return '\n'.join(lines)
# ──────────────────────────────────────────────────────────────────────
# Actions
# ──────────────────────────────────────────────────────────────────────
def action_override(self):
"""Manager-only override of a detected conflict. Requires justification."""
self.ensure_one()
if not self.env.user.has_group('activeblue_familylaw.group_admin'):
raise UserError(_(
'Only a Family Law Administrator can override a conflict of interest.'
))
if not (self.override_justification or '').strip():
raise UserError(_(
'A written justification is required to override a conflict.'
))
self.write({
'status': 'override',
'override_user_id': self.env.user.id,
'override_date': fields.Datetime.now(),
})
self.case_id.conflict_check_passed = True
self.case_id.message_post(
body=(
f'<strong>⚠ CONFLICT OVERRIDE</strong> authorized by '
f'{self.env.user.name}.<br/>'
f'<b>Justification:</b> {self.override_justification}'
),
subtype_xmlid='mail.mt_comment',
)
return True

View File

@@ -78,6 +78,9 @@ access_fl_fee_waiver_petitioner,fl.fee.waiver petitioner,model_fl_fee_waiver,gro
access_fl_income_withholding_admin,fl.income.withholding admin,model_fl_income_withholding,group_admin,1,1,1,1
access_fl_income_withholding_paralegal,fl.income.withholding paralegal,model_fl_income_withholding,group_paralegal,1,1,1,0
access_fl_income_withholding_petitioner,fl.income.withholding petitioner,model_fl_income_withholding,group_portal_petitioner,1,0,0,0
# ── fl.conflict.check ────────────────────────────────────────────────────────
access_fl_conflict_check_admin,fl.conflict.check admin,model_fl_conflict_check,group_admin,1,1,1,1
access_fl_conflict_check_paralegal,fl.conflict.check paralegal,model_fl_conflict_check,group_paralegal,1,1,1,0
# ── fl.intake.wizard ─────────────────────────────────────────────────────────
access_fl_intake_wizard_admin,fl.intake.wizard admin,model_fl_intake_wizard,group_admin,1,1,1,1
access_fl_intake_wizard_paralegal,fl.intake.wizard paralegal,model_fl_intake_wizard,group_paralegal,1,1,1,1
1 id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
78 access_fl_income_withholding_admin,fl.income.withholding admin,model_fl_income_withholding,group_admin,1,1,1,1
79 access_fl_income_withholding_paralegal,fl.income.withholding paralegal,model_fl_income_withholding,group_paralegal,1,1,1,0
80 access_fl_income_withholding_petitioner,fl.income.withholding petitioner,model_fl_income_withholding,group_portal_petitioner,1,0,0,0
81 # ── fl.conflict.check ────────────────────────────────────────────────────────
82 access_fl_conflict_check_admin,fl.conflict.check admin,model_fl_conflict_check,group_admin,1,1,1,1
83 access_fl_conflict_check_paralegal,fl.conflict.check paralegal,model_fl_conflict_check,group_paralegal,1,1,1,0
84 # ── fl.intake.wizard ─────────────────────────────────────────────────────────
85 access_fl_intake_wizard_admin,fl.intake.wizard admin,model_fl_intake_wizard,group_admin,1,1,1,1
86 access_fl_intake_wizard_paralegal,fl.intake.wizard paralegal,model_fl_intake_wizard,group_paralegal,1,1,1,1

View File

@@ -18,6 +18,9 @@
<button name="action_open_support_calculator" string="Open Calculator"
type="object"
attrs="{'invisible': [('case_type', 'not in', ['modification','dissolution_children','paternity','custody_modification'])]}"/>
<button name="action_run_conflict_check" string="Run Conflict Check"
type="object"
groups="activeblue_familylaw.group_admin,activeblue_familylaw.group_paralegal"/>
</header>
<!-- Attorney Referral Banner -->
@@ -29,6 +32,18 @@
<br/>Legal Services of Greater Miami: (305) 576-0080
</div>
<!-- Conflict of Interest Banner -->
<div class="alert alert-danger" role="alert"
attrs="{'invisible': ['|', ('conflict_check_passed', '=', True), ('conflict_check_id', '=', False)]}">
<strong>🚩 CONFLICT OF INTEREST — UNRESOLVED</strong>
— A conflict check has flagged matching parties on other open cases.
This case cannot advance past Intake until the conflict is overridden by an administrator.
Open the
<button name="action_run_conflict_check" type="object"
class="btn btn-link p-0 align-baseline" string="conflict check"/>
for details.
</div>
<!-- DV Safety Note -->
<field name="dv_safety_note" widget="html"
attrs="{'invisible': [('domestic_violence_flag', '=', False)]}"/>
@@ -97,10 +112,16 @@
<field name="respondent_answered"/>
</group>
</group>
<group string="Safety Flags">
<field name="domestic_violence_flag"/>
<field name="dv_injunction_active"
attrs="{'invisible': [('domestic_violence_flag', '=', False)]}"/>
<group>
<group string="Safety Flags">
<field name="domestic_violence_flag"/>
<field name="dv_injunction_active"
attrs="{'invisible': [('domestic_violence_flag', '=', False)]}"/>
</group>
<group string="Conflict of Interest">
<field name="conflict_check_passed" readonly="1"/>
<field name="conflict_check_id" readonly="1"/>
</group>
</group>
<separator string="Party Details (Income, Employment, Service)"/>
<field name="party_ids">
@@ -375,6 +396,8 @@
<field name="attorney_referral_flag"/>
<field name="domestic_violence_flag"/>
<field name="overdue_deadline_count"/>
<field name="conflict_check_passed"/>
<field name="conflict_check_id"/>
<field name="petitioner_id"/>
<templates>
<t t-name="kanban-card">
@@ -404,6 +427,9 @@
<t t-esc="record.overdue_deadline_count.raw_value"/> Overdue
</span>
</t>
<t t-if="record.conflict_check_id.raw_value &amp;&amp; !record.conflict_check_passed.raw_value">
<span class="badge rounded-pill text-bg-danger ms-1">Conflict</span>
</t>
</div>
<div class="o_kanban_record_bottom" t-if="record.next_deadline.raw_value">
<div class="oe_kanban_bottom_left text-muted small">

View File

@@ -0,0 +1,119 @@
<?xml version="1.0" encoding="utf-8"?>
<odoo>
<data>
<!-- ══════════════════════════════════════════════════════
FORM VIEW
══════════════════════════════════════════════════════ -->
<record id="view_fl_conflict_check_form" model="ir.ui.view">
<field name="name">fl.conflict.check.form</field>
<field name="model">fl.conflict.check</field>
<field name="arch" type="xml">
<form string="Conflict of Interest Check">
<header>
<button name="action_override" string="Override Conflict"
type="object" class="oe_highlight"
confirm="Override this conflict of interest? This allows the case to advance past Intake. A justification is required."
attrs="{'invisible': [('status', '!=', 'conflict')]}"
groups="activeblue_familylaw.group_admin"/>
<field name="status" widget="statusbar"
statusbar_visible="clear,conflict,override"/>
</header>
<sheet>
<div class="alert alert-success" role="alert"
attrs="{'invisible': [('status', '!=', 'clear')]}">
<strong>✅ CLEAR</strong> — No conflicting parties found on other open cases.
</div>
<div class="alert alert-danger" role="alert"
attrs="{'invisible': [('status', '!=', 'conflict')]}">
<strong>🚩 CONFLICT DETECTED</strong> — One or more parties match
parties on other open cases. The case cannot advance past Intake
until this is overridden by an administrator.
</div>
<div class="alert alert-warning" role="alert"
attrs="{'invisible': [('status', '!=', 'override')]}">
<strong>⚠ OVERRIDDEN</strong> — This conflict was overridden by
<field name="override_user_id" readonly="1" nolabel="1" class="oe_inline"/>
on <field name="override_date" readonly="1" nolabel="1" class="oe_inline"/>.
</div>
<group>
<group>
<field name="case_id"/>
<field name="check_date"/>
</group>
<group>
<field name="matched_case_ids" widget="many2many_tags"
attrs="{'invisible': [('matched_case_ids', '=', [])]}"/>
</group>
</group>
<separator string="Match Detail"/>
<field name="match_detail" nolabel="1" readonly="1"/>
<separator string="Override Justification"
attrs="{'invisible': [('status', '=', 'clear')]}"/>
<field name="override_justification" nolabel="1"
placeholder="Required to override a conflict — explain why the firm may proceed despite the match."
attrs="{'invisible': [('status', '=', 'clear')], 'readonly': [('status', '=', 'override')]}"/>
</sheet>
<div class="oe_chatter">
<field name="message_follower_ids"/>
<field name="message_ids"/>
</div>
</form>
</field>
</record>
<!-- ══════════════════════════════════════════════════════
TREE VIEW
══════════════════════════════════════════════════════ -->
<record id="view_fl_conflict_check_tree" model="ir.ui.view">
<field name="name">fl.conflict.check.tree</field>
<field name="model">fl.conflict.check</field>
<field name="arch" type="xml">
<tree string="Conflict Checks"
decoration-danger="status == 'conflict'"
decoration-warning="status == 'override'"
decoration-success="status == 'clear'">
<field name="case_id"/>
<field name="check_date"/>
<field name="status"/>
<field name="matched_case_ids" widget="many2many_tags"/>
</tree>
</field>
</record>
<!-- ══════════════════════════════════════════════════════
SEARCH VIEW
══════════════════════════════════════════════════════ -->
<record id="view_fl_conflict_check_search" model="ir.ui.view">
<field name="name">fl.conflict.check.search</field>
<field name="model">fl.conflict.check</field>
<field name="arch" type="xml">
<search string="Search Conflict Checks">
<field name="case_id"/>
<filter string="Conflicts" name="conflicts"
domain="[('status', '=', 'conflict')]"/>
<filter string="Overridden" name="overridden"
domain="[('status', '=', 'override')]"/>
<filter string="Clear" name="clear"
domain="[('status', '=', 'clear')]"/>
<group expand="0" string="Group By">
<filter string="Status" name="group_status"
context="{'group_by': 'status'}"/>
</group>
</search>
</field>
</record>
<!-- ══════════════════════════════════════════════════════
ACTION
══════════════════════════════════════════════════════ -->
<record id="action_fl_conflict_check_list" model="ir.actions.act_window">
<field name="name">Conflict Checks</field>
<field name="res_model">fl.conflict.check</field>
<field name="view_mode">tree,form</field>
<field name="search_view_id" ref="view_fl_conflict_check_search"/>
<field name="context">{'search_default_conflicts': 1}</field>
</record>
</data>
</odoo>

View File

@@ -87,6 +87,13 @@
action="action_fl_discovery_list"
sequence="60"/>
<menuitem
id="menu_fl_conflict_checks"
name="Conflict Checks"
parent="menu_fl_cases"
action="action_fl_conflict_check_list"
sequence="70"/>
<!-- ══════════════════════════════════════════════════════
SUPPORT CALCULATOR SUB-MENU
══════════════════════════════════════════════════════ -->