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>
This commit is contained in:
221
addons/google_translate/models/google_translate.go
Normal file
221
addons/google_translate/models/google_translate.go
Normal file
@@ -0,0 +1,221 @@
|
||||
package models
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"os"
|
||||
|
||||
"odoo-go/pkg/orm"
|
||||
"odoo-go/pkg/tools"
|
||||
)
|
||||
|
||||
var translateClient *tools.APIClient
|
||||
|
||||
func getTranslateClient() *tools.APIClient {
|
||||
if translateClient != nil {
|
||||
return translateClient
|
||||
}
|
||||
apiKey := os.Getenv("GOOGLE_TRANSLATE_API_KEY")
|
||||
if apiKey == "" {
|
||||
return nil
|
||||
}
|
||||
translateClient = tools.NewAPIClient("https://translation.googleapis.com", apiKey)
|
||||
return translateClient
|
||||
}
|
||||
|
||||
func initGoogleTranslate() {
|
||||
// Register a translation model for storing translations + providing RPC methods
|
||||
m := orm.NewModel("google.translate", orm.ModelOpts{
|
||||
Description: "Google Translation Service",
|
||||
Type: orm.ModelTransient, // No persistent table needed
|
||||
})
|
||||
|
||||
m.AddFields(
|
||||
orm.Text("source_text", orm.FieldOpts{String: "Source Text"}),
|
||||
orm.Char("source_lang", orm.FieldOpts{String: "Source Language", Default: "auto"}),
|
||||
orm.Char("target_lang", orm.FieldOpts{String: "Target Language", Default: "de"}),
|
||||
orm.Text("translated_text", orm.FieldOpts{String: "Translated Text"}),
|
||||
)
|
||||
|
||||
// translate: Translate text from one language to another
|
||||
// Usage via RPC: call_kw("google.translate", "translate", [args])
|
||||
m.RegisterMethod("translate", func(rs *orm.Recordset, args ...interface{}) (interface{}, error) {
|
||||
client := getTranslateClient()
|
||||
if client == nil {
|
||||
return nil, fmt.Errorf("google_translate: GOOGLE_TRANSLATE_API_KEY not configured")
|
||||
}
|
||||
|
||||
text := ""
|
||||
targetLang := "de"
|
||||
sourceLang := ""
|
||||
|
||||
if len(args) > 0 {
|
||||
text, _ = args[0].(string)
|
||||
}
|
||||
if len(args) > 1 {
|
||||
targetLang, _ = args[1].(string)
|
||||
}
|
||||
if len(args) > 2 {
|
||||
sourceLang, _ = args[2].(string)
|
||||
}
|
||||
|
||||
if text == "" {
|
||||
return nil, fmt.Errorf("google_translate: no text provided")
|
||||
}
|
||||
|
||||
params := map[string]string{
|
||||
"q": text,
|
||||
"target": targetLang,
|
||||
"format": "text",
|
||||
}
|
||||
if sourceLang != "" && sourceLang != "auto" {
|
||||
params["source"] = sourceLang
|
||||
}
|
||||
|
||||
var result TranslateResponse
|
||||
err := client.GetJSON("/language/translate/v2", params, &result)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("google_translate: API error: %w", err)
|
||||
}
|
||||
|
||||
if len(result.Data.Translations) == 0 {
|
||||
return nil, fmt.Errorf("google_translate: no translation returned")
|
||||
}
|
||||
|
||||
t := result.Data.Translations[0]
|
||||
return map[string]interface{}{
|
||||
"translated_text": t.TranslatedText,
|
||||
"detected_source": t.DetectedSourceLanguage,
|
||||
}, nil
|
||||
})
|
||||
|
||||
// translate_batch: Translate multiple texts at once
|
||||
m.RegisterMethod("translate_batch", func(rs *orm.Recordset, args ...interface{}) (interface{}, error) {
|
||||
client := getTranslateClient()
|
||||
if client == nil {
|
||||
return nil, fmt.Errorf("google_translate: GOOGLE_TRANSLATE_API_KEY not configured")
|
||||
}
|
||||
|
||||
texts, ok := args[0].([]interface{})
|
||||
if !ok || len(texts) == 0 {
|
||||
return nil, fmt.Errorf("google_translate: texts array required")
|
||||
}
|
||||
targetLang := "de"
|
||||
if len(args) > 1 {
|
||||
targetLang, _ = args[1].(string)
|
||||
}
|
||||
|
||||
var results []map[string]interface{}
|
||||
for _, t := range texts {
|
||||
text, _ := t.(string)
|
||||
if text == "" {
|
||||
continue
|
||||
}
|
||||
|
||||
var result TranslateResponse
|
||||
err := client.GetJSON("/language/translate/v2", map[string]string{
|
||||
"q": text,
|
||||
"target": targetLang,
|
||||
"format": "text",
|
||||
}, &result)
|
||||
|
||||
if err != nil || len(result.Data.Translations) == 0 {
|
||||
results = append(results, map[string]interface{}{
|
||||
"source": text, "translated": text, "error": fmt.Sprintf("%v", err),
|
||||
})
|
||||
continue
|
||||
}
|
||||
|
||||
results = append(results, map[string]interface{}{
|
||||
"source": text,
|
||||
"translated": result.Data.Translations[0].TranslatedText,
|
||||
"detected": result.Data.Translations[0].DetectedSourceLanguage,
|
||||
})
|
||||
}
|
||||
|
||||
return results, nil
|
||||
})
|
||||
|
||||
// detect_language: Detect the language of a text
|
||||
m.RegisterMethod("detect_language", func(rs *orm.Recordset, args ...interface{}) (interface{}, error) {
|
||||
client := getTranslateClient()
|
||||
if client == nil {
|
||||
return nil, fmt.Errorf("google_translate: GOOGLE_TRANSLATE_API_KEY not configured")
|
||||
}
|
||||
|
||||
text := ""
|
||||
if len(args) > 0 {
|
||||
text, _ = args[0].(string)
|
||||
}
|
||||
|
||||
var result DetectResponse
|
||||
err := client.GetJSON("/language/translate/v2/detect", map[string]string{
|
||||
"q": text,
|
||||
}, &result)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if len(result.Data.Detections) > 0 && len(result.Data.Detections[0]) > 0 {
|
||||
d := result.Data.Detections[0][0]
|
||||
return map[string]interface{}{
|
||||
"language": d.Language,
|
||||
"confidence": d.Confidence,
|
||||
}, nil
|
||||
}
|
||||
return nil, fmt.Errorf("google_translate: language detection failed")
|
||||
})
|
||||
|
||||
// supported_languages: List supported languages
|
||||
m.RegisterMethod("supported_languages", func(rs *orm.Recordset, args ...interface{}) (interface{}, error) {
|
||||
client := getTranslateClient()
|
||||
if client == nil {
|
||||
return nil, fmt.Errorf("google_translate: GOOGLE_TRANSLATE_API_KEY not configured")
|
||||
}
|
||||
|
||||
var result LanguagesResponse
|
||||
err := client.GetJSON("/language/translate/v2/languages", map[string]string{
|
||||
"target": "de",
|
||||
}, &result)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var langs []map[string]string
|
||||
for _, l := range result.Data.Languages {
|
||||
langs = append(langs, map[string]string{
|
||||
"code": l.Language,
|
||||
"name": l.Name,
|
||||
})
|
||||
}
|
||||
return langs, nil
|
||||
})
|
||||
}
|
||||
|
||||
// --- Google Translate API Response Types ---
|
||||
|
||||
type TranslateResponse struct {
|
||||
Data struct {
|
||||
Translations []struct {
|
||||
TranslatedText string `json:"translatedText"`
|
||||
DetectedSourceLanguage string `json:"detectedSourceLanguage"`
|
||||
} `json:"translations"`
|
||||
} `json:"data"`
|
||||
}
|
||||
|
||||
type DetectResponse struct {
|
||||
Data struct {
|
||||
Detections [][]struct {
|
||||
Language string `json:"language"`
|
||||
Confidence float64 `json:"confidence"`
|
||||
} `json:"detections"`
|
||||
} `json:"data"`
|
||||
}
|
||||
|
||||
type LanguagesResponse struct {
|
||||
Data struct {
|
||||
Languages []struct {
|
||||
Language string `json:"language"`
|
||||
Name string `json:"name"`
|
||||
} `json:"languages"`
|
||||
} `json:"data"`
|
||||
}
|
||||
5
addons/google_translate/models/init.go
Normal file
5
addons/google_translate/models/init.go
Normal file
@@ -0,0 +1,5 @@
|
||||
package models
|
||||
|
||||
func Init() {
|
||||
initGoogleTranslate()
|
||||
}
|
||||
27
addons/google_translate/module.go
Normal file
27
addons/google_translate/module.go
Normal file
@@ -0,0 +1,27 @@
|
||||
// Package google_translate provides Google Cloud Translation integration.
|
||||
// OPT-IN: Only active when GOOGLE_TRANSLATE_API_KEY is configured.
|
||||
//
|
||||
// Features:
|
||||
// - Translate any text field on any record
|
||||
// - Auto-detect source language
|
||||
// - Batch translation support
|
||||
package google_translate
|
||||
|
||||
import (
|
||||
"odoo-go/addons/google_translate/models"
|
||||
"odoo-go/pkg/modules"
|
||||
)
|
||||
|
||||
func init() {
|
||||
modules.Register(&modules.Module{
|
||||
Name: "google_translate",
|
||||
Description: "Google Cloud Translation",
|
||||
Version: "19.0.1.0.0",
|
||||
Category: "Integration",
|
||||
Depends: []string{"base"},
|
||||
Application: false,
|
||||
Installable: true,
|
||||
Sequence: 100,
|
||||
Init: models.Init,
|
||||
})
|
||||
}
|
||||
Reference in New Issue
Block a user