mirror of
https://github.com/ManyakRus/crud_generator.git
synced 2024-12-22 00:36:41 +02:00
214 lines
5.3 KiB
Go
214 lines
5.3 KiB
Go
|
package schema
|
||
|
|
||
|
import (
|
||
|
"database/sql"
|
||
|
"fmt"
|
||
|
)
|
||
|
|
||
|
// https://github.com/golang/go/issues/7408
|
||
|
//
|
||
|
// https://github.com/golang/go/issues/7408#issuecomment-252046876
|
||
|
//
|
||
|
// If this package were to be part of database/sql, then the API would become like:-
|
||
|
//
|
||
|
// func (db *DB) Table(schema, table string) ([]*ColumnType, error)
|
||
|
// func (db *DB) TableNames() ([][2]string, error)
|
||
|
// func (db *DB) Tables() (map[[2]string][]*ColumnType, error)
|
||
|
// func (db *DB) View(schema, view string) ([]*ColumnType, error)
|
||
|
// func (db *DB) ViewNames() ([][2]string, error)
|
||
|
// func (db *DB) Views() (map[[2]string][]*ColumnType, error)
|
||
|
//
|
||
|
|
||
|
//
|
||
|
|
||
|
// UnknownDriverError is returned when there is no matching
|
||
|
// database driver type name in the driverDialect table.
|
||
|
//
|
||
|
// Errors of this kind are caused by using an unsupported
|
||
|
// database driver/dialect, or if/when a database driver
|
||
|
// developer renames the type underlying calls to db.Driver().
|
||
|
type UnknownDriverError struct {
|
||
|
Driver string
|
||
|
}
|
||
|
|
||
|
// Error returns a formatted string description.
|
||
|
func (e UnknownDriverError) Error() string {
|
||
|
return fmt.Sprintf("unknown database driver: %s", e.Driver)
|
||
|
}
|
||
|
|
||
|
//
|
||
|
|
||
|
// Tables returns column type metadata for all tables in the current schema.
|
||
|
//
|
||
|
// The returned map is keyed by table name tuples.
|
||
|
func Tables(db *sql.DB) (map[[2]string][]*sql.ColumnType, error) {
|
||
|
d, err := getDialect(db)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
names, err := d.TableNames(db)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if len(names) == 0 {
|
||
|
return nil, nil
|
||
|
}
|
||
|
m := make(map[[2]string][]*sql.ColumnType, len(names))
|
||
|
for _, n := range names {
|
||
|
ct, err := d.ColumnTypes(db, n[0], n[1])
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
m[n] = ct
|
||
|
}
|
||
|
return m, nil
|
||
|
}
|
||
|
|
||
|
// Views returns column type metadata for all views in the current schema.
|
||
|
//
|
||
|
// The returned map is keyed by view name tuples.
|
||
|
func Views(db *sql.DB) (map[[2]string][]*sql.ColumnType, error) {
|
||
|
d, err := getDialect(db)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
names, err := d.ViewNames(db)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if len(names) == 0 {
|
||
|
return nil, nil
|
||
|
}
|
||
|
m := make(map[[2]string][]*sql.ColumnType, len(names))
|
||
|
for _, n := range names {
|
||
|
ct, err := d.ColumnTypes(db, n[0], n[1])
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
m[n] = ct
|
||
|
}
|
||
|
return m, nil
|
||
|
}
|
||
|
|
||
|
// TableNames returns a list of all table names.
|
||
|
//
|
||
|
// Each name consists of a [2]string tuple: schema name, table name.
|
||
|
func TableNames(db *sql.DB) ([][2]string, error) {
|
||
|
d, err := getDialect(db)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return d.TableNames(db)
|
||
|
}
|
||
|
|
||
|
// ViewNames returns a list of all view names.
|
||
|
//
|
||
|
// Each name consists of a [2]string tuple: schema name, view name.
|
||
|
func ViewNames(db *sql.DB) ([][2]string, error) {
|
||
|
d, err := getDialect(db)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return d.ViewNames(db)
|
||
|
}
|
||
|
|
||
|
// ColumnTypes returns the column type metadata for the given object (table or view) in the given schema.
|
||
|
//
|
||
|
// Setting schema to an empty string results in the current schema being used.
|
||
|
func ColumnTypes(db *sql.DB, schema, object string) ([]*sql.ColumnType, error) {
|
||
|
d, err := getDialect(db)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return d.ColumnTypes(db, schema, object)
|
||
|
}
|
||
|
|
||
|
// PrimaryKey returns a list of column names making up the primary
|
||
|
// key for the given table in the given schema.
|
||
|
func PrimaryKey(db *sql.DB, schema, table string) ([]string, error) {
|
||
|
d, err := getDialect(db)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return d.PrimaryKey(db, schema, table)
|
||
|
}
|
||
|
|
||
|
// fetchNames executes the given query with an optional name parameter,
|
||
|
// and returns a list of table/view/column names.
|
||
|
//
|
||
|
// The name parameter (if not "") is passed as a parameter to db.Query.
|
||
|
func fetchNames(db *sql.DB, query, schema, name string) ([]string, error) {
|
||
|
var rows *sql.Rows
|
||
|
var err error
|
||
|
if len(schema) > 0 {
|
||
|
rows, err = db.Query(query, schema, name)
|
||
|
} else {
|
||
|
rows, err = db.Query(query, name)
|
||
|
}
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
defer rows.Close()
|
||
|
// Scan result into list of names.
|
||
|
var names []string
|
||
|
n := ""
|
||
|
for rows.Next() {
|
||
|
err = rows.Scan(&n)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
names = append(names, n)
|
||
|
}
|
||
|
return names, nil
|
||
|
}
|
||
|
|
||
|
// fetchObjectNames executes the given query
|
||
|
// and returns a list of table/view/column names.
|
||
|
func fetchObjectNames(db *sql.DB, query string) ([][2]string, error) {
|
||
|
var rows *sql.Rows
|
||
|
var err error
|
||
|
rows, err = db.Query(query)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
defer rows.Close()
|
||
|
// Scan result into list of names.
|
||
|
var names [][2]string
|
||
|
s := ""
|
||
|
n := ""
|
||
|
for rows.Next() {
|
||
|
err = rows.Scan(&s, &n)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
names = append(names, [2]string{s, n})
|
||
|
}
|
||
|
return names, nil
|
||
|
}
|
||
|
|
||
|
func getDialect(db *sql.DB) (dialect, error) {
|
||
|
dt := fmt.Sprintf("%T", db.Driver())
|
||
|
d, ok := driverDialect[dt]
|
||
|
if !ok {
|
||
|
return nil, UnknownDriverError{Driver: dt}
|
||
|
}
|
||
|
return d, nil
|
||
|
}
|
||
|
|
||
|
// fetchColumnTypes queries the database and returns column's type metadata
|
||
|
// for a single table or view.
|
||
|
func fetchColumnTypes(db *sql.DB, query, schema, name string, escapeIdent func(string) string) ([]*sql.ColumnType, error) {
|
||
|
if schema == "" {
|
||
|
query = fmt.Sprintf(query, escapeIdent(name))
|
||
|
} else {
|
||
|
n := fmt.Sprintf("%s.%s", escapeIdent(schema), escapeIdent(name))
|
||
|
query = fmt.Sprintf(query, n)
|
||
|
}
|
||
|
rows, err := db.Query(query)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
defer rows.Close()
|
||
|
return rows.ColumnTypes()
|
||
|
}
|