mirror of
https://github.com/casdoor/casdoor.git
synced 2025-05-22 18:25:47 +08:00
235 lines
6.3 KiB
Go
235 lines
6.3 KiB
Go
// Copyright 2022 The Casdoor Authors. All Rights Reserved.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package object
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/casdoor/casdoor/conf"
|
|
"github.com/casdoor/casdoor/util"
|
|
xormadapter "github.com/casdoor/xorm-adapter/v3"
|
|
"github.com/xorm-io/core"
|
|
"github.com/xorm-io/xorm"
|
|
)
|
|
|
|
type Adapter struct {
|
|
Owner string `xorm:"varchar(100) notnull pk" json:"owner"`
|
|
Name string `xorm:"varchar(100) notnull pk" json:"name"`
|
|
CreatedTime string `xorm:"varchar(100)" json:"createdTime"`
|
|
|
|
Table string `xorm:"varchar(100)" json:"table"`
|
|
UseSameDb bool `json:"useSameDb"`
|
|
Type string `xorm:"varchar(100)" json:"type"`
|
|
DatabaseType string `xorm:"varchar(100)" json:"databaseType"`
|
|
Host string `xorm:"varchar(100)" json:"host"`
|
|
Port int `json:"port"`
|
|
User string `xorm:"varchar(100)" json:"user"`
|
|
Password string `xorm:"varchar(100)" json:"password"`
|
|
Database string `xorm:"varchar(100)" json:"database"`
|
|
|
|
*xormadapter.Adapter `xorm:"-" json:"-"`
|
|
}
|
|
|
|
func GetAdapterCount(owner, field, value string) (int64, error) {
|
|
session := GetSession(owner, -1, -1, field, value, "", "")
|
|
return session.Count(&Adapter{})
|
|
}
|
|
|
|
func GetAdapters(owner string) ([]*Adapter, error) {
|
|
adapters := []*Adapter{}
|
|
err := ormer.Engine.Desc("created_time").Find(&adapters, &Adapter{Owner: owner})
|
|
if err != nil {
|
|
return adapters, err
|
|
}
|
|
|
|
return adapters, nil
|
|
}
|
|
|
|
func GetPaginationAdapters(owner string, offset, limit int, field, value, sortField, sortOrder string) ([]*Adapter, error) {
|
|
adapters := []*Adapter{}
|
|
session := GetSession(owner, offset, limit, field, value, sortField, sortOrder)
|
|
err := session.Find(&adapters)
|
|
if err != nil {
|
|
return adapters, err
|
|
}
|
|
|
|
return adapters, nil
|
|
}
|
|
|
|
func getAdapter(owner, name string) (*Adapter, error) {
|
|
if owner == "" || name == "" {
|
|
return nil, nil
|
|
}
|
|
|
|
adapter := Adapter{Owner: owner, Name: name}
|
|
existed, err := ormer.Engine.Get(&adapter)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if existed {
|
|
return &adapter, nil
|
|
} else {
|
|
return nil, nil
|
|
}
|
|
}
|
|
|
|
func GetAdapter(id string) (*Adapter, error) {
|
|
owner, name := util.GetOwnerAndNameFromId(id)
|
|
return getAdapter(owner, name)
|
|
}
|
|
|
|
func UpdateAdapter(id string, adapter *Adapter) (bool, error) {
|
|
owner, name := util.GetOwnerAndNameFromId(id)
|
|
if adapter, err := getAdapter(owner, name); adapter == nil {
|
|
return false, err
|
|
}
|
|
|
|
if name != adapter.Name {
|
|
err := adapterChangeTrigger(name, adapter.Name)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
|
|
session := ormer.Engine.ID(core.PK{owner, name}).AllCols()
|
|
if adapter.Password == "***" {
|
|
session.Omit("password")
|
|
}
|
|
affected, err := session.Update(adapter)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
return affected != 0, nil
|
|
}
|
|
|
|
func AddAdapter(adapter *Adapter) (bool, error) {
|
|
affected, err := ormer.Engine.Insert(adapter)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
return affected != 0, nil
|
|
}
|
|
|
|
func DeleteAdapter(adapter *Adapter) (bool, error) {
|
|
affected, err := ormer.Engine.ID(core.PK{adapter.Owner, adapter.Name}).Delete(&Adapter{})
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
return affected != 0, nil
|
|
}
|
|
|
|
func (adapter *Adapter) GetId() string {
|
|
return fmt.Sprintf("%s/%s", adapter.Owner, adapter.Name)
|
|
}
|
|
|
|
func (adapter *Adapter) InitAdapter() error {
|
|
if adapter.Adapter != nil {
|
|
return nil
|
|
}
|
|
|
|
var driverName string
|
|
var dataSourceName string
|
|
if adapter.UseSameDb || adapter.isBuiltIn() {
|
|
driverName = conf.GetConfigString("driverName")
|
|
dataSourceName = conf.GetConfigString("dataSourceName")
|
|
if conf.GetConfigString("driverName") == "mysql" {
|
|
dataSourceName = dataSourceName + conf.GetConfigString("dbName")
|
|
}
|
|
} else {
|
|
driverName = adapter.DatabaseType
|
|
switch driverName {
|
|
case "mssql":
|
|
dataSourceName = fmt.Sprintf("sqlserver://%s:%s@%s:%d?database=%s", adapter.User,
|
|
adapter.Password, adapter.Host, adapter.Port, adapter.Database)
|
|
case "mysql":
|
|
dataSourceName = fmt.Sprintf("%s:%s@tcp(%s:%d)/%s", adapter.User,
|
|
adapter.Password, adapter.Host, adapter.Port, adapter.Database)
|
|
case "postgres":
|
|
dataSourceName = fmt.Sprintf("user=%s password=%s host=%s port=%d sslmode=disable dbname=%s", adapter.User,
|
|
adapter.Password, adapter.Host, adapter.Port, adapter.Database)
|
|
case "CockroachDB":
|
|
dataSourceName = fmt.Sprintf("user=%s password=%s host=%s port=%d sslmode=disable dbname=%s serial_normalization=virtual_sequence",
|
|
adapter.User, adapter.Password, adapter.Host, adapter.Port, adapter.Database)
|
|
case "sqlite3":
|
|
dataSourceName = fmt.Sprintf("file:%s", adapter.Host)
|
|
default:
|
|
return fmt.Errorf("unsupported database type: %s", adapter.DatabaseType)
|
|
}
|
|
}
|
|
|
|
if !isCloudIntranet {
|
|
dataSourceName = strings.ReplaceAll(dataSourceName, "dbi.", "db.")
|
|
}
|
|
|
|
engine, err := xorm.NewEngine(driverName, dataSourceName)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if (adapter.UseSameDb || adapter.isBuiltIn()) && driverName == "postgres" {
|
|
schema := util.GetValueFromDataSourceName("search_path", dataSourceName)
|
|
if schema != "" {
|
|
engine.SetSchema(schema)
|
|
}
|
|
}
|
|
|
|
var tableName string
|
|
if driverName == "mssql" {
|
|
tableName = fmt.Sprintf("[%s]", adapter.Table)
|
|
} else {
|
|
tableName = adapter.Table
|
|
}
|
|
|
|
adapter.Adapter, err = xormadapter.NewAdapterByEngineWithTableName(engine, tableName, "")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func adapterChangeTrigger(oldName string, newName string) error {
|
|
session := ormer.Engine.NewSession()
|
|
defer session.Close()
|
|
|
|
err := session.Begin()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
enforcer := new(Enforcer)
|
|
enforcer.Adapter = newName
|
|
_, err = session.Where("adapter=?", oldName).Update(enforcer)
|
|
if err != nil {
|
|
session.Rollback()
|
|
return err
|
|
}
|
|
|
|
return session.Commit()
|
|
}
|
|
|
|
func (adapter *Adapter) isBuiltIn() bool {
|
|
if adapter.Owner != "built-in" {
|
|
return false
|
|
}
|
|
|
|
return adapter.Name == "user-adapter-built-in" || adapter.Name == "api-adapter-built-in"
|
|
}
|