ORM: - SQL Constraints support (Model.AddSQLConstraint, applied in InitDatabase) - Translatable field Read (ir_translation lookup for non-en_US) - active_test filter in SearchCount + ReadGroup (consistency with Search) - Environment.Ref() improved (format validation, parameterized query) Server: - /web/action/run endpoint (server action execution stub) - /web/model/get_definitions (field metadata for multiple models) - Binary field serving rewritten: reads from DB, falls back to SVG with record initial (fixes avatar/logo rendering) Business modules deepened: - Account: action_post validation (partner, lines), sequence numbering (JOURNAL/YYYY/NNNN), action_register_payment, remove_move_reconcile - Sale: action_cancel, action_draft, action_view_invoice - Purchase: button_draft - Stock: action_cancel on picking - CRM: action_set_won_rainbowman, convert_opportunity - HR: hr.contract model (employee, wage, dates, state) - Project: action_blocked, task stage seed data Views: - Cancel/Reset buttons in sale.form header - Register Payment button in invoice.form (visible when posted+unpaid) Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
95 lines
2.5 KiB
Go
95 lines
2.5 KiB
Go
package server
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"odoo-go/pkg/orm"
|
|
)
|
|
|
|
// handleImage serves images for model records.
|
|
// Mirrors: odoo/addons/web/controllers/binary.py content_image()
|
|
//
|
|
// The client requests images like:
|
|
// /web/image/res.partner/5/avatar_128
|
|
// /web/image/product.product/1/image_1920
|
|
//
|
|
// We first try to read the binary field from the database.
|
|
// If no data is found, we fall back to an SVG placeholder.
|
|
func (s *Server) handleImage(w http.ResponseWriter, r *http.Request) {
|
|
// Parse: /web/image/<model>/<id>/<field>
|
|
parts := strings.Split(strings.TrimPrefix(r.URL.Path, "/web/image/"), "/")
|
|
|
|
model := ""
|
|
id := int64(0)
|
|
field := "image_1920"
|
|
|
|
// Also accept query parameters (legacy format)
|
|
if qm := r.URL.Query().Get("model"); qm != "" {
|
|
model = qm
|
|
}
|
|
if qf := r.URL.Query().Get("field"); qf != "" {
|
|
field = qf
|
|
}
|
|
|
|
if len(parts) >= 1 && parts[0] != "" {
|
|
model = parts[0]
|
|
}
|
|
if len(parts) >= 2 {
|
|
if n, err := strconv.ParseInt(parts[1], 10, 64); err == nil {
|
|
id = n
|
|
}
|
|
}
|
|
if len(parts) >= 3 && parts[2] != "" {
|
|
field = parts[2]
|
|
}
|
|
|
|
// Try to read binary data from DB
|
|
if model != "" && id > 0 {
|
|
m := orm.Registry.Get(model)
|
|
if m != nil {
|
|
f := m.GetField(field)
|
|
if f != nil && f.Type == orm.TypeBinary {
|
|
table := m.Table()
|
|
var data []byte
|
|
ctx := r.Context()
|
|
_ = s.pool.QueryRow(ctx,
|
|
fmt.Sprintf(`SELECT "%s" FROM "%s" WHERE id = $1`, f.Column(), table), id,
|
|
).Scan(&data)
|
|
if len(data) > 0 {
|
|
// Detect content type
|
|
contentType := http.DetectContentType(data)
|
|
w.Header().Set("Content-Type", contentType)
|
|
w.Header().Set("Cache-Control", "public, max-age=604800")
|
|
w.Write(data)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Fallback: SVG placeholder with the record's initial letter
|
|
initial := "?"
|
|
if model != "" && id > 0 {
|
|
m := orm.Registry.Get(model)
|
|
if m != nil {
|
|
var name string
|
|
_ = s.pool.QueryRow(r.Context(),
|
|
fmt.Sprintf(`SELECT COALESCE(name, '') FROM "%s" WHERE id = $1`, m.Table()), id,
|
|
).Scan(&name)
|
|
if len(name) > 0 {
|
|
initial = strings.ToUpper(name[:1])
|
|
}
|
|
}
|
|
}
|
|
|
|
w.Header().Set("Content-Type", "image/svg+xml")
|
|
w.Header().Set("Cache-Control", "public, max-age=604800")
|
|
fmt.Fprintf(w, `<svg xmlns="http://www.w3.org/2000/svg" width="64" height="64" viewBox="0 0 64 64">
|
|
<rect width="64" height="64" rx="32" fill="#71639e"/>
|
|
<text x="32" y="40" font-family="sans-serif" font-size="28" fill="white" text-anchor="middle">%s</text>
|
|
</svg>`, initial)
|
|
}
|