Files
goodie/pkg/modules/module.go
Marc 0ed29fe2fd Odoo ERP ported to Go — complete backend + original OWL frontend
Full port of Odoo's ERP system from Python to Go, with the original
Odoo JavaScript frontend (OWL framework) running against the Go server.

Backend (10,691 LoC Go):
- Custom ORM: CRUD, domains→SQL with JOINs, computed fields, sequences
- 93 models across 14 modules (base, account, sale, stock, purchase, hr,
  project, crm, fleet, product, l10n_de, google_address/translate/calendar)
- Auth with bcrypt + session cookies
- Setup wizard (company, SKR03 chart, admin, demo data)
- Double-entry bookkeeping constraint
- Sale→Invoice workflow (confirm SO → generate invoice → post)
- SKR03 chart of accounts (110 accounts) + German taxes (USt/VSt)
- Record rules (multi-company filter)
- Google integrations as opt-in modules (Maps, Translate, Calendar)

Frontend:
- Odoo's original OWL webclient (503 JS modules, 378 XML templates)
- JS transpiled via Odoo's js_transpiler (ES modules → odoo.define)
- SCSS compiled to CSS (675KB) via dart-sass
- XML templates compiled to registerTemplate() JS calls
- Static file serving from Odoo source addons
- Login page, session management, menu navigation
- Contacts list view renders with real data from PostgreSQL

Infrastructure:
- 14MB single binary (CGO_ENABLED=0)
- Docker Compose (Go server + PostgreSQL 16)
- Zero phone-home (no outbound calls to odoo.com)

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-31 01:45:09 +02:00

77 lines
2.1 KiB
Go

// Package modules implements Odoo's module system.
// Mirrors: odoo/modules/module.py, odoo/modules/loading.py
package modules
import (
"fmt"
"sync"
)
// Module represents an Odoo addon module.
// Mirrors: ir.module.module + __manifest__.py
type Module struct {
Name string // Technical name (e.g., "base", "account", "sale")
Description string // Human-readable description
Version string // Module version
Category string // Module category
Depends []string // Required modules (dependency list)
AutoInstall bool // Auto-install when all depends are installed
Application bool // Show in app list
Installable bool // Can be installed
Sequence int // Loading order
// Init function registers models, fields, and methods.
// Called during module loading in dependency order.
Init func()
// PostInit is called after all modules are loaded.
PostInit func()
// Data files to load (SQL seed data, etc.)
Data []string
}
// ModuleRegistry holds all registered modules.
// Mirrors: odoo/modules/module.py loaded modules
var ModuleRegistry = &moduleRegistry{
modules: make(map[string]*Module),
}
type moduleRegistry struct {
mu sync.RWMutex
modules map[string]*Module
order []string // Registration order
}
// Register adds a module to the registry.
func Register(m *Module) {
ModuleRegistry.mu.Lock()
defer ModuleRegistry.mu.Unlock()
if m.Name == "" {
panic("modules: module name is required")
}
if _, exists := ModuleRegistry.modules[m.Name]; exists {
panic(fmt.Sprintf("modules: module %q already registered", m.Name))
}
ModuleRegistry.modules[m.Name] = m
ModuleRegistry.order = append(ModuleRegistry.order, m.Name)
}
// Get returns a module by name.
func Get(name string) *Module {
ModuleRegistry.mu.RLock()
defer ModuleRegistry.mu.RUnlock()
return ModuleRegistry.modules[name]
}
// All returns all registered module names.
func All() []string {
ModuleRegistry.mu.RLock()
defer ModuleRegistry.mu.RUnlock()
result := make([]string, len(ModuleRegistry.order))
copy(result, ModuleRegistry.order)
return result
}