- Portal: /my/* routes, signup, password reset, portal user support - Email Inbound: IMAP polling (go-imap/v2), thread matching - Discuss: mail.channel, long-polling bus, DM, unread count - Cron: ir.cron runner (goroutine scheduler) - Bank Import, CSV/Excel Import - Automation (ir.actions.server) - Fetchmail service - HR Payroll model - Various fixes across account, sale, stock, purchase, crm, hr, project Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
205 lines
7.0 KiB
Go
205 lines
7.0 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",
|
|
Inherits: map[string]string{"res.partner": "partner_id"},
|
|
})
|
|
|
|
// -- 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)",
|
|
}),
|
|
orm.Char("signup_token", orm.FieldOpts{String: "Signup Token"}),
|
|
orm.Datetime("signup_expiration", orm.FieldOpts{String: "Signup Token Expiration"}),
|
|
)
|
|
|
|
// _compute_share: portal/public users have share=true (not in group_user).
|
|
// Mirrors: odoo/addons/base/models/res_users.py Users._compute_share()
|
|
m.RegisterMethod("_compute_share", func(rs *orm.Recordset, args ...interface{}) (interface{}, error) {
|
|
env := rs.Env()
|
|
// Look up group_user ID
|
|
var groupUserID int64
|
|
err := env.Tx().QueryRow(env.Ctx(),
|
|
`SELECT g.id FROM res_groups g
|
|
JOIN ir_model_data imd ON imd.res_id = g.id AND imd.model = 'res.groups'
|
|
WHERE imd.module = 'base' AND imd.name = 'group_user'`).Scan(&groupUserID)
|
|
if err != nil {
|
|
return nil, nil // Can't determine, skip
|
|
}
|
|
|
|
for _, id := range rs.IDs() {
|
|
var inGroup bool
|
|
err := env.Tx().QueryRow(env.Ctx(),
|
|
`SELECT EXISTS(
|
|
SELECT 1 FROM res_groups_res_users_rel
|
|
WHERE res_groups_id = $1 AND res_users_id = $2
|
|
)`, groupUserID, id).Scan(&inGroup)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
env.Model("res.users").Browse(id).Write(orm.Values{"share": !inGroup})
|
|
}
|
|
return nil, nil
|
|
})
|
|
|
|
// -- 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"}),
|
|
)
|
|
}
|