Antigravity-awesome-skills odoo-security-rules
Expert in Odoo access control: ir.model.access.csv, record rules (ir.rule), groups, and multi-company security patterns.
install
source · Clone the upstream repo
git clone https://github.com/sickn33/antigravity-awesome-skills
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/sickn33/antigravity-awesome-skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/plugins/antigravity-awesome-skills-claude/skills/odoo-security-rules" ~/.claude/skills/sickn33-antigravity-awesome-skills-odoo-security-rules && rm -rf "$T"
manifest:
plugins/antigravity-awesome-skills-claude/skills/odoo-security-rules/SKILL.mdsource content
Odoo Security Rules
Overview
Security in Odoo is managed at two levels: model-level access (who can read/write which models) and record-level rules (which records a user can see). This skill helps you write correct
ir.model.access.csv entries and ir.rule domain-based record rules.
When to Use This Skill
- Setting up access rights for a new custom module.
- Restricting records so users only see their own data or their company's data.
- Debugging "Access Denied" or "You are not allowed to access" errors.
- Implementing multi-company record visibility rules.
How It Works
- Activate: Mention
and describe the access scenario.@odoo-security-rules - Generate: Get correct CSV access lines and XML record rules.
- Debug: Paste an access error and get a diagnosis with the fix.
Examples
Example 1: ir.model.access.csv
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink access_hospital_patient_user,hospital.patient.user,model_hospital_patient,base.group_user,1,0,0,0 access_hospital_patient_manager,hospital.patient.manager,model_hospital_patient,base.group_erp_manager,1,1,1,1
Note: Use
for ERP managers, notbase.group_erp_manager— that group is reserved for Odoo's technical superusers. Always create a custom group for module-specific manager roles:base.group_system<record id="group_hospital_manager" model="res.groups"> <field name="name">Hospital Manager</field> <field name="category_id" ref="base.module_category_hidden"/> </record>
Example 2: Record Rule — Users See Only Their Own Records
<record id="rule_hospital_patient_own" model="ir.rule"> <field name="name">Hospital Patient: Own Records Only</field> <field name="model_id" ref="model_hospital_patient"/> <field name="domain_force">[('create_uid', '=', user.id)]</field> <field name="groups" eval="[(4, ref('base.group_user'))]"/> <field name="perm_read" eval="True"/> <field name="perm_write" eval="True"/> <field name="perm_create" eval="True"/> <field name="perm_unlink" eval="False"/> </record>
Important: If you omit
, the rule becomes global and applies to ALL users, including admins. Always assign a group unless you explicitly intend a global restriction.<field name="groups">
Example 3: Multi-Company Record Rule
<record id="rule_hospital_patient_company" model="ir.rule"> <field name="name">Hospital Patient: Multi-Company</field> <field name="model_id" ref="model_hospital_patient"/> <field name="domain_force"> ['|', ('company_id', '=', False), ('company_id', 'in', company_ids)] </field> <field name="groups" eval="[(4, ref('base.group_user'))]"/> </record>
Best Practices
- ✅ Do: Start with the most restrictive access and open up as needed.
- ✅ Do: Use
(plural) in multi-company rules — it includes all companies the user belongs to.company_ids - ✅ Do: Test rules using a non-admin user in debug mode —
bypasses all record rules entirely.sudo() - ✅ Do: Create dedicated security groups per module rather than reusing core Odoo groups.
- ❌ Don't: Give
to regular users unless deletion is explicitly required by the business process.perm_unlink = 1 - ❌ Don't: Leave
blank ingroup_id
unless you intend to grant public (unauthenticated) access.ir.model.access.csv - ❌ Don't: Use
for module managers — that grants full technical access including server configurations.base.group_system
Limitations
- Does not cover field-level access control (
read/write restrictions) — those require custom OWL or Python overrides.ir.model.fields - Portal and public user access rules have additional nuances not fully covered here; test carefully with
.base.group_portal - Record rules are bypassed by
— any code running in superuser context ignores allsudo()
entries.ir.rule - Does not cover row-level security via PostgreSQL (RLS) — Odoo manages all security at the ORM layer.