oef/orm/orm.go

122 lines
2.7 KiB
Go
Raw Normal View History

2019-11-04 15:00:46 +01:00
package orm
import (
"fmt"
"net/http"
"path"
"reflect"
"strings"
"git.andreafazzi.eu/andrea/oef/config"
2019-11-04 15:00:46 +01:00
"github.com/jinzhu/gorm"
"github.com/jinzhu/inflection"
_ "github.com/jinzhu/gorm/dialects/mysql"
)
type IDer interface {
GetID() uint
}
type GetFn func(map[string]string, http.ResponseWriter, *http.Request) (interface{}, error)
2019-11-04 15:00:46 +01:00
2020-01-14 16:28:27 +01:00
type Database struct {
Config *config.ConfigT
2019-11-04 15:00:46 +01:00
2020-01-14 16:28:27 +01:00
_db *gorm.DB
fns map[string]func(map[string]string, http.ResponseWriter, *http.Request) (interface{}, error)
2019-11-04 15:00:46 +01:00
}
2020-01-14 16:28:27 +01:00
func NewDatabase(config *config.ConfigT) (*Database, error) {
db := new(Database)
db.fns = make(db*gorm.DB, map[string]func(map[string]string, http.ResponseWriter, *http.Request) (interface{}, error), 0)
db._db, err := gorm.Open("mysql", fmt.Sprintf("%s?%s", config.Config.Orm.Connection, config.Config.Orm.Options))
2019-11-04 15:00:46 +01:00
if err != nil {
return nil, err
}
2020-01-14 16:28:27 +01:00
return db
2019-11-04 15:00:46 +01:00
}
2020-01-14 16:28:27 +01:00
func (db *Database) AutoMigrate(models ...interface{}) {
if err := db._db.AutoMigrate(models...).Error; err != nil {
2019-11-04 15:00:46 +01:00
panic(err)
}
}
2019-12-09 08:27:46 +01:00
func CreateCategories() {
2019-12-09 11:15:42 +01:00
for _, name := range categories {
2019-12-09 08:27:46 +01:00
var category Category
if err := currDB.FirstOrCreate(&category, Category{Name: name}).Error; err != nil {
panic(err)
}
}
}
2019-11-04 15:00:46 +01:00
func Use(db *gorm.DB) {
currDB = db
}
func DB() *gorm.DB {
return currDB
}
func MapHandlers(models []interface{}) error {
for _, model := range models {
2020-01-02 13:01:21 +01:00
name := inflection.Plural(strings.ToLower(ModelName(model)))
2019-11-04 15:00:46 +01:00
for p, action := range map[string]string{
"": "ReadAll",
"create/": "Create",
"{id}": "Read",
"{id}/update": "Update",
"{id}/delete": "Delete",
} {
method := reflect.ValueOf(model).MethodByName(action)
if !method.IsValid() {
return fmt.Errorf("Action %s is not defined for model %s", action, name)
}
joinedPath := path.Join("/", name, p)
if strings.HasSuffix(p, "/") {
joinedPath += "/"
}
fns[joinedPath] = method.Interface().(func(map[string]string, http.ResponseWriter, *http.Request) (interface{}, error))
2019-11-04 15:00:46 +01:00
}
}
return nil
}
func GetFunc(path string) (GetFn, error) {
fn, ok := fns[path]
if !ok {
return nil, fmt.Errorf("Can't map path %s to any model methods.", path)
}
return fn, nil
}
func GetNothing(args map[string]string) (interface{}, error) {
return nil, nil
}
func PostNothing(args map[string]string, w http.ResponseWriter, r *http.Request) (IDer, error) {
2019-11-04 15:00:46 +01:00
return nil, nil
}
2020-01-02 13:01:21 +01:00
func ModelName(s interface{}) string {
2020-01-07 18:33:11 +01:00
t := reflect.TypeOf(s)
switch t.Kind() {
case reflect.Ptr:
elem := t.Elem()
if strings.Contains(elem.String(), "[]") {
return strings.Replace(elem.String(), "[]*orm.", "", -1)
}
return elem.Name()
case reflect.Slice:
return strings.Replace(t.Elem().String(), "*orm.", "", -1)
default:
2019-11-04 15:00:46 +01:00
return t.Name()
}
}