mirror of
https://github.com/bitechdev/ResolveSpec.git
synced 2025-11-13 09:53:53 +00:00
135 lines
3.6 KiB
Go
135 lines
3.6 KiB
Go
package modelregistry
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"sync"
|
|
)
|
|
|
|
// DefaultModelRegistry implements ModelRegistry interface
|
|
type DefaultModelRegistry struct {
|
|
models map[string]interface{}
|
|
mutex sync.RWMutex
|
|
}
|
|
|
|
// Global default registry instance
|
|
var defaultRegistry = &DefaultModelRegistry{
|
|
models: make(map[string]interface{}),
|
|
}
|
|
|
|
// NewModelRegistry creates a new model registry
|
|
func NewModelRegistry() *DefaultModelRegistry {
|
|
return &DefaultModelRegistry{
|
|
models: make(map[string]interface{}),
|
|
}
|
|
}
|
|
|
|
func (r *DefaultModelRegistry) RegisterModel(name string, model interface{}) error {
|
|
r.mutex.Lock()
|
|
defer r.mutex.Unlock()
|
|
|
|
if _, exists := r.models[name]; exists {
|
|
return fmt.Errorf("model %s already registered", name)
|
|
}
|
|
|
|
// Validate that model is a non-pointer struct
|
|
modelType := reflect.TypeOf(model)
|
|
if modelType == nil {
|
|
return fmt.Errorf("model cannot be nil")
|
|
}
|
|
|
|
originalType := modelType
|
|
|
|
// Unwrap pointers, slices, and arrays to check the underlying type
|
|
for modelType.Kind() == reflect.Ptr || modelType.Kind() == reflect.Slice || modelType.Kind() == reflect.Array {
|
|
modelType = modelType.Elem()
|
|
}
|
|
|
|
// Validate that the underlying type is a struct
|
|
if modelType.Kind() != reflect.Struct {
|
|
return fmt.Errorf("model must be a struct or pointer to struct, got %s", originalType.String())
|
|
}
|
|
|
|
// If a pointer/slice/array was passed, unwrap to the base struct
|
|
if originalType != modelType {
|
|
// Create a zero value of the struct type
|
|
model = reflect.New(modelType).Elem().Interface()
|
|
}
|
|
|
|
// Additional check: ensure model is not a pointer
|
|
finalType := reflect.TypeOf(model)
|
|
if finalType.Kind() == reflect.Ptr {
|
|
return fmt.Errorf("model must be a non-pointer struct, got pointer to %s. Use MyModel{} instead of &MyModel{}", finalType.Elem().Name())
|
|
}
|
|
|
|
r.models[name] = model
|
|
return nil
|
|
}
|
|
|
|
func (r *DefaultModelRegistry) GetModel(name string) (interface{}, error) {
|
|
r.mutex.RLock()
|
|
defer r.mutex.RUnlock()
|
|
|
|
model, exists := r.models[name]
|
|
if !exists {
|
|
return nil, fmt.Errorf("model %s not found", name)
|
|
}
|
|
|
|
return model, nil
|
|
}
|
|
|
|
func (r *DefaultModelRegistry) GetAllModels() map[string]interface{} {
|
|
r.mutex.RLock()
|
|
defer r.mutex.RUnlock()
|
|
|
|
result := make(map[string]interface{})
|
|
for k, v := range r.models {
|
|
result[k] = v
|
|
}
|
|
return result
|
|
}
|
|
|
|
func (r *DefaultModelRegistry) GetModelByEntity(schema, entity string) (interface{}, error) {
|
|
// Try full name first
|
|
fullName := fmt.Sprintf("%s.%s", schema, entity)
|
|
if model, err := r.GetModel(fullName); err == nil {
|
|
return model, nil
|
|
}
|
|
|
|
// Fallback to entity name only
|
|
return r.GetModel(entity)
|
|
}
|
|
|
|
// Global convenience functions using the default registry
|
|
|
|
// RegisterModel registers a model with the default global registry
|
|
func RegisterModel(model interface{}, name string) error {
|
|
return defaultRegistry.RegisterModel(name, model)
|
|
}
|
|
|
|
// GetModelByName retrieves a model from the default global registry by name
|
|
func GetModelByName(name string) (interface{}, error) {
|
|
return defaultRegistry.GetModel(name)
|
|
}
|
|
|
|
// IterateModels iterates over all models in the default global registry
|
|
func IterateModels(fn func(name string, model interface{})) {
|
|
defaultRegistry.mutex.RLock()
|
|
defer defaultRegistry.mutex.RUnlock()
|
|
|
|
for name, model := range defaultRegistry.models {
|
|
fn(name, model)
|
|
}
|
|
}
|
|
|
|
// GetModels returns a list of all models in the default global registry
|
|
func GetModels() []interface{} {
|
|
defaultRegistry.mutex.RLock()
|
|
defer defaultRegistry.mutex.RUnlock()
|
|
|
|
models := make([]interface{}, 0, len(defaultRegistry.models))
|
|
for _, model := range defaultRegistry.models {
|
|
models = append(models, model)
|
|
}
|
|
return models
|
|
} |