Files
goodie/addons/base/models/res_users.go
Marc b57176de2f Bring odoo-go to ~70%: read_group, record rules, admin, sessions
Phase 1: read_group/web_read_group with SQL GROUP BY, aggregates
  (sum/avg/min/max/count/array_agg/sum_currency), date granularity,
  M2O groupby resolution to [id, display_name].

Phase 2: Record rules with domain_force parsing (Python literal parser),
  global AND + group OR merging. Domain operators: child_of, parent_of,
  any, not any compiled to SQL hierarchy/EXISTS queries.

Phase 3: Button dispatch via /web/dataset/call_button, method return
  values interpreted as actions. Payment register wizard
  (account.payment.register) for sale→invoice→pay flow.

Phase 4: ir.filters, ir.default, product fields expanded, SO line
  product_id onchange, ir_model+ir_model_fields DB seeding.

Phase 5: CSV export (/web/export/csv), attachment upload/download
  via ir.attachment, fields_get with aggregator hints.

Admin/System: Session persistence (PostgreSQL-backed), ir.config_parameter
  with get_param/set_param, ir.cron, ir.logging, res.lang, res.config.settings
  with company-related fields, Settings form view. Technical menu with
  Views/Actions/Parameters/Security/Logging sub-menus. User change_password,
  preferences. Password never exposed in UI/API.

Bugfixes: false→nil for varchar/int fields, int32 in toInt64, call_button
  route with trailing slash, create_invoices returns action, search view
  always included, get_formview_action, name_create, ir.http stub.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-02 19:26:08 +02:00

173 lines
5.8 KiB
Go

package models
import (
"fmt"
"odoo-go/pkg/orm"
"odoo-go/pkg/tools"
)
// initResUsers registers the res.users model.
// Mirrors: odoo/addons/base/models/res_users.py class Users
//
// In Odoo, res.users inherits from res.partner via _inherits.
// Every user has a linked partner record for contact info.
func initResUsers() {
m := orm.NewModel("res.users", orm.ModelOpts{
Description: "Users",
Order: "login",
})
// -- Authentication --
m.AddFields(
orm.Char("login", orm.FieldOpts{String: "Login", Required: true, Index: true}),
orm.Char("password", orm.FieldOpts{String: "Password"}),
orm.Boolean("active", orm.FieldOpts{String: "Active", Default: true}),
)
// -- Partner link (Odoo: _inherits = {'res.partner': 'partner_id'}) --
m.AddFields(
orm.Many2one("partner_id", "res.partner", orm.FieldOpts{
String: "Related Partner", Required: true, OnDelete: orm.OnDeleteRestrict,
}),
orm.Char("name", orm.FieldOpts{String: "Name", Related: "partner_id.name"}),
orm.Char("email", orm.FieldOpts{String: "Email", Related: "partner_id.email"}),
)
// -- Company --
m.AddFields(
orm.Many2one("company_id", "res.company", orm.FieldOpts{
String: "Company", Required: true, Index: true,
}),
orm.Many2many("company_ids", "res.company", orm.FieldOpts{String: "Allowed Companies"}),
)
// -- Groups / Permissions --
m.AddFields(
orm.Many2many("groups_id", "res.groups", orm.FieldOpts{String: "Groups"}),
)
// -- Preferences --
m.AddFields(
orm.Char("lang", orm.FieldOpts{String: "Language", Default: "en_US"}),
orm.Char("tz", orm.FieldOpts{String: "Timezone", Default: "UTC"}),
orm.Selection("notification_type", []orm.SelectionItem{
{Value: "email", Label: "Handle by Emails"},
{Value: "inbox", Label: "Handle in Odoo"},
}, orm.FieldOpts{String: "Notification", Default: "email"}),
orm.Binary("image_1920", orm.FieldOpts{String: "Avatar"}),
orm.Char("signature", orm.FieldOpts{String: "Email Signature"}),
)
// -- Status --
m.AddFields(
orm.Boolean("share", orm.FieldOpts{
String: "Share User", Compute: "_compute_share", Store: true,
Help: "External user with limited access (portal/public)",
}),
)
// -- Methods --
// action_get returns the "Change My Preferences" action for the current user.
// Mirrors: odoo/addons/base/models/res_users.py Users.action_get()
m.RegisterMethod("action_get", func(rs *orm.Recordset, args ...interface{}) (interface{}, error) {
return map[string]interface{}{
"type": "ir.actions.act_window",
"name": "Change My Preferences",
"res_model": "res.users",
"view_mode": "form",
"views": [][]interface{}{{false, "form"}},
"target": "new",
"res_id": rs.Env().UID(),
"context": map[string]interface{}{},
}, nil
})
// change_password: verifies old password and sets a new one.
// Mirrors: odoo/addons/base/models/res_users.py Users.change_password()
m.RegisterMethod("change_password", func(rs *orm.Recordset, args ...interface{}) (interface{}, error) {
if len(args) < 2 {
return false, fmt.Errorf("change_password requires old_password and new_password")
}
oldPw, _ := args[0].(string)
newPw, _ := args[1].(string)
env := rs.Env()
uid := env.UID()
// Verify old password
var hashedPw string
err := env.Tx().QueryRow(env.Ctx(),
`SELECT password FROM res_users WHERE id = $1`, uid).Scan(&hashedPw)
if err != nil {
return false, fmt.Errorf("user not found")
}
if !tools.CheckPassword(hashedPw, oldPw) {
return false, fmt.Errorf("incorrect old password")
}
// Hash and set new password
newHash, err := tools.HashPassword(newPw)
if err != nil {
return false, err
}
_, err = env.Tx().Exec(env.Ctx(),
`UPDATE res_users SET password = $1 WHERE id = $2`, newHash, uid)
if err != nil {
return false, err
}
return true, nil
})
// preference_save: called when saving user preferences.
// Mirrors: odoo/addons/base/models/res_users.py Users.preference_save()
m.RegisterMethod("preference_save", func(rs *orm.Recordset, args ...interface{}) (interface{}, error) {
// Preferences are saved via normal write; just return true.
return true, nil
})
// preference_change_password: alias for change_password from preferences dialog.
// Mirrors: odoo/addons/base/models/res_users.py Users.preference_change_password()
m.RegisterMethod("preference_change_password", func(rs *orm.Recordset, args ...interface{}) (interface{}, error) {
return rs.Env().Model("res.users").Browse(rs.Env().UID()).Read([]string{"id"})
})
}
// initResGroups registers the res.groups model.
// Mirrors: odoo/addons/base/models/res_users.py class Groups
//
// Groups define permission sets. Users belong to groups.
// Groups can imply other groups (hierarchy).
func initResGroups() {
m := orm.NewModel("res.groups", orm.ModelOpts{
Description: "Access Groups",
Order: "name",
})
m.AddFields(
orm.Char("name", orm.FieldOpts{String: "Name", Required: true, Translate: true}),
orm.Text("comment", orm.FieldOpts{String: "Comment"}),
orm.Many2one("category_id", "ir.module.category", orm.FieldOpts{String: "Application"}),
orm.Char("color", orm.FieldOpts{String: "Color Index"}),
orm.Char("full_name", orm.FieldOpts{String: "Group Name", Compute: "_compute_full_name"}),
orm.Boolean("share", orm.FieldOpts{String: "Share Group", Default: false}),
)
// -- Relationships --
m.AddFields(
orm.Many2many("users", "res.users", orm.FieldOpts{String: "Users"}),
orm.Many2many("implied_ids", "res.groups", orm.FieldOpts{
String: "Inherits",
Help: "Users of this group automatically inherit those groups",
}),
)
// -- Access Control --
m.AddFields(
orm.One2many("model_access", "ir.model.access", "group_id", orm.FieldOpts{String: "Access Controls"}),
orm.One2many("rule_groups", "ir.rule", "group_id", orm.FieldOpts{String: "Rules"}),
orm.Many2many("menu_access", "ir.ui.menu", orm.FieldOpts{String: "Access Menu"}),
)
}