package base import ( "database/sql" "log" "reflect" "unicode" "github.com/samber/do" "github.com/samber/lo" "git.fsdpf.net/go/contracts" "git.fsdpf.net/go/contracts/res_type" "git.fsdpf.net/go/db" "git.fsdpf.net/go/queue" ) // 资源 type Resource struct { container *do.Injector Uuid string `db:"uuid"` PUuid string `db:"pUuid"` Code string `db:"code"` Name string `db:"name"` IsResVirtual bool `db:"isVirtual"` Table string `db:"table"` Namespace string `db:"namespace"` Workspace string `db:"workspace"` Primarykey string `db:"primaryKey"` IsHistoryRecord bool `db:"isHistoryRecord"` HistoryCacheMax int `db:"historyCacheMax"` Fields ResFields `db:"fields"` Roles res_type.ResFieldByMap `db:"roles"` UpdatedAt string `db:"updated_at"` CreatedAt string `db:"created_at"` } // 资源UUID func (this Resource) GetUuid() string { return this.Uuid } // 资源CODE func (this Resource) GetCode() string { return this.Code } // 资源名 func (this Resource) GetName() string { return this.Name } // 主键 func (this Resource) GetPrimarykey() string { return this.Primarykey } // 是否虚拟资源 func (this Resource) IsVirtual() bool { return this.IsResVirtual } // 是否系统资源 func (this Resource) IsSystem() bool { return this.Namespace == "Framework\\Service" } // 资源字段 func (this Resource) GetFields() (result []contracts.ResField) { for _, item := range this.Fields { result = append(result, item) } return result } // 资源字段 func (this Resource) GetField(code string) (contracts.ResField, bool) { return lo.Find(this.GetFields(), func(v contracts.ResField) bool { return v.GetCode() == code }) } // 判断资源字段 func (this Resource) HasField(code string) bool { return lo.SomeBy(this.GetFields(), func(v contracts.ResField) bool { return v.GetCode() == code }) } // 开启事物 func (this Resource) BeginTransaction() (*db.Transaction, error) { return this.GetDB().Connection.BeginTransaction() } // 获取资源对应的数据库连接 func (this Resource) GetDB() *db.Builder { db := do.MustInvoke[db.DB](this.container) if this.IsSystem() { return db.Connection("service-support").Query() } return db.Connection("default").Query() } // 获取资源对应的数据表 func (this Resource) GetTable() db.Expression { if this.IsVirtual() { return db.Raw("(" + this.Table + ")") } return db.Raw(this.Table) } func (this Resource) GetDBDriver() string { return this.GetDB().Connection.GetConfig().Driver } func (this Resource) GetAuthDBTable(u contracts.User, params ...any) *db.Builder { this.GetRolesCondition(u) return this.GetDBTable(append(params, u)...) } // GetDBTable("Test", contracts.User) func (this Resource) GetDBTable(params ...any) *db.Builder { builder := this.GetDB() var user contracts.User alias := this.Code for _, param := range params { switch v := param.(type) { case *db.Transaction: builder.Tx = v case string: alias = v case contracts.User: user = v } } // 格式化数据库存储数据 builder.Before(func(b *db.Builder, t string, data ...map[string]any) { if t == db.TYPE_UPDATE { // 格式化保存数据 this.formatSaveValue(data[0]) } if t == db.TYPE_INSERT { // 移除 table alias b.Table(string(this.GetTable())) for i := 0; i < len(data); i++ { // 格式化保存数据 this.formatSaveValue(data[i]) // 填充保存数据 this.fillSaveValue(data[i], user, db.TYPE_INSERT) } } }) // 资源事件 this.onResEvent(builder) // 用户事件 if this.IsHistoryRecord { this.onUserEvent(builder, user) } // 虚拟资源暂时不考虑鉴权 if !this.IsVirtual() { // 返回鉴权后的 DB Builder // return } return builder.Table(string(this.GetTable()), alias) } func (this Resource) WithRolesCondition(b *db.Builder, roles ...string) { } // 获取鉴权条件 func (this Resource) GetRolesCondition(u contracts.User) { // isFullRight := false // isFullNot := false // roles := do.MustInvoke[GetResRoles](this.container)(this.GetUuid()) // GetResRelations := do.MustInvoke[GetResRelations](this.container) // subTables := lo.Reduce(roles, func(carry *db.Builder, item ResRole, _ int) *db.Builder { // db := this.GetDB().Table(string(this.GetTable()), this.GetCode()).Select("`" + this.GetCode() + "`.*") // joins := lo.Filter(GetResRelations(item.Uuid), func(item ResRelation, _ int) bool { // return item.Type == "inner" || item.Type == "left" || item.Type == "right" // }) // join := orm.NewJoin(contracts.RelationType(item.Type), item.Code, oResource, item.RelationResource, item.RelationField, item.RelationForeignKey) // // 关联扩展条件 // join.SetCondition(orm.NewConditionByRes(GetResConditions(item.Uuid))) // return carry // }, nil) // fmt.Println(subTables.ToSql()) } // 格式化保存数据 func (this Resource) formatSaveValue(data map[string]any) { // for k, v := range data { if k == "id" || k == "created_user" || k == "created_at" || k == "deleted_at" || k == "updated_at" { delete(data, k) } else if val, ok := v.(db.Expression); ok { data[k] = val } else if field, ok := this.GetField(k); ok { data[k] = field.ToValue(v) } } } // 填充保存数据 func (this Resource) fillSaveValue(data map[string]any, u contracts.User, t string) { for _, field := range this.GetFields() { fCode := field.GetCode() if fCode == "id" || fCode == "created_user" || fCode == "created_at" || fCode == "deleted_at" || fCode == "updated_at" || fCode == "owned_user" { continue } // 只有新增默认字段 if _, ok := data[fCode]; !ok { data[fCode] = field.GetRawDefault(this.GetDBDriver()) } } // 拥有者 if _, ok := data["owned_user"]; !ok { data["owned_user"] = u.Uuid() } // 创建者 data["created_user"] = u.Uuid() if this.GetDBDriver() == "sqlite" { // 更新时间 // sqlite 不能自动更新时间, "DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP" data["updated_at"] = db.Raw("CURRENT_TIMESTAMP") } else { } } func (this Resource) GetStruct(extends ...reflect.StructField) any { fields := []reflect.StructField{} for _, field := range this.Fields { if unicode.IsLetter(rune(field.Code[0])) { fields = append(fields, field.ToStructField()) } else { log.Printf("资源字段错误, 必须以字母开头 <- %s", field.Code) } } fields = lo.UniqBy(append(fields, extends...), func(v reflect.StructField) string { return v.Name }) t := reflect.StructOf(fields) return reflect.New(t).Interface() } func (this Resource) GetSliceStruct(extends ...reflect.StructField) any { t := reflect.TypeOf(this.GetStruct(extends...)) st := reflect.SliceOf(t.Elem()) return reflect.New(st).Interface() } // 资源事件 func (this Resource) onResEvent(builder *db.Builder) { builder.After(func(b *db.Builder, t string, result sql.Result, err error, data ...map[string]any) { if err != nil || t == db.TYPE_SELECT { return } else if num, err := result.RowsAffected(); num == 0 || err != nil { return } // 全局触发器 // 1. 清除系统缓存 if err := do.MustInvoke[contracts.Queue](this.container).Publish(contracts.ResQueueTopic, contracts.ResQueuePayload{ Type: t, Res: this, Result: result, }); err != nil { log.Println("Queue Publish Err:", contracts.ResQueueTopic, err) } }) } // 用户事件 func (this Resource) onUserEvent(builder *db.Builder, user contracts.User) { old := []map[string]any{} builder.Before(func(b *db.Builder, t string, data ...map[string]any) { if t != db.TYPE_UPDATE && t != db.TYPE_DELETE { return } // 查询保存之前的数据 if _, err := b.Get(&old); err != nil { panic(err) } }) builder.After(func(b *db.Builder, t string, result sql.Result, err error, data ...map[string]any) { if err != nil || t == db.TYPE_SELECT { return } else if num, err := result.RowsAffected(); num == 0 || err != nil { return } if user == nil { user = GetAnonymous() } // 触发消息队列 if err := do.MustInvoke[contracts.Queue](this.container).Publish(contracts.UserQueuetTopic, contracts.UserQueuePayload{ Type: t, User: user, Res: this, Old: old, New: data, Result: result, }); err != nil { log.Println("Queue Publish Err:", contracts.UserQueuetTopic, err) } }) }