2022-07-06 23:19:05 +02:00
|
|
|
package daos
|
|
|
|
|
|
|
|
import (
|
2023-02-18 19:33:42 +02:00
|
|
|
"fmt"
|
|
|
|
|
2022-07-06 23:19:05 +02:00
|
|
|
"github.com/pocketbase/dbx"
|
2023-02-18 19:33:42 +02:00
|
|
|
"github.com/pocketbase/pocketbase/models"
|
2022-07-06 23:19:05 +02:00
|
|
|
)
|
|
|
|
|
2023-02-21 20:03:27 +02:00
|
|
|
// HasTable checks if a table (or view) with the provided name exists (case insensitive).
|
2022-07-06 23:19:05 +02:00
|
|
|
func (dao *Dao) HasTable(tableName string) bool {
|
|
|
|
var exists bool
|
|
|
|
|
|
|
|
err := dao.DB().Select("count(*)").
|
|
|
|
From("sqlite_schema").
|
2023-02-21 20:03:27 +02:00
|
|
|
AndWhere(dbx.HashExp{"type": []any{"table", "view"}}).
|
2022-07-06 23:19:05 +02:00
|
|
|
AndWhere(dbx.NewExp("LOWER([[name]])=LOWER({:tableName})", dbx.Params{"tableName": tableName})).
|
|
|
|
Limit(1).
|
|
|
|
Row(&exists)
|
|
|
|
|
|
|
|
return err == nil && exists
|
|
|
|
}
|
|
|
|
|
2023-03-22 17:15:17 +02:00
|
|
|
// TableColumns returns all column names of a single table by its name.
|
|
|
|
func (dao *Dao) TableColumns(tableName string) ([]string, error) {
|
2022-07-06 23:19:05 +02:00
|
|
|
columns := []string{}
|
|
|
|
|
|
|
|
err := dao.DB().NewQuery("SELECT name FROM PRAGMA_TABLE_INFO({:tableName})").
|
|
|
|
Bind(dbx.Params{"tableName": tableName}).
|
|
|
|
Column(&columns)
|
|
|
|
|
|
|
|
return columns, err
|
|
|
|
}
|
|
|
|
|
2023-03-22 17:15:17 +02:00
|
|
|
// TableInfo returns the `table_info` pragma result for the specified table.
|
|
|
|
func (dao *Dao) TableInfo(tableName string) ([]*models.TableInfoRow, error) {
|
2023-02-18 19:33:42 +02:00
|
|
|
info := []*models.TableInfoRow{}
|
|
|
|
|
|
|
|
err := dao.DB().NewQuery("SELECT * FROM PRAGMA_TABLE_INFO({:tableName})").
|
|
|
|
Bind(dbx.Params{"tableName": tableName}).
|
|
|
|
All(&info)
|
2023-02-18 20:31:41 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-02-18 19:33:42 +02:00
|
|
|
|
2023-02-18 20:31:41 +02:00
|
|
|
// mattn/go-sqlite3 doesn't throw an error on invalid or missing table
|
|
|
|
// so we additionally have to check whether the loaded info result is nonempty
|
|
|
|
if len(info) == 0 {
|
|
|
|
return nil, fmt.Errorf("empty table info probably due to invalid or missing table %s", tableName)
|
|
|
|
}
|
|
|
|
|
|
|
|
return info, nil
|
2023-02-18 19:33:42 +02:00
|
|
|
}
|
|
|
|
|
2023-03-21 15:31:20 +02:00
|
|
|
// TableIndexes returns a name grouped map with all non empty index of the specified table.
|
|
|
|
//
|
|
|
|
// Note: This method doesn't return an error on nonexisting table.
|
|
|
|
func (dao *Dao) TableIndexes(tableName string) (map[string]string, error) {
|
|
|
|
indexes := []struct {
|
|
|
|
Name string
|
|
|
|
Sql string
|
|
|
|
}{}
|
|
|
|
|
|
|
|
err := dao.DB().Select("name", "sql").
|
|
|
|
From("sqlite_master").
|
|
|
|
AndWhere(dbx.NewExp("sql is not null")).
|
|
|
|
AndWhere(dbx.HashExp{
|
|
|
|
"type": "index",
|
|
|
|
"tbl_name": tableName,
|
|
|
|
}).
|
|
|
|
All(&indexes)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
result := make(map[string]string, len(indexes))
|
|
|
|
|
|
|
|
for _, idx := range indexes {
|
|
|
|
result[idx.Name] = idx.Sql
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
2022-07-06 23:19:05 +02:00
|
|
|
// DeleteTable drops the specified table.
|
2023-02-18 19:33:42 +02:00
|
|
|
//
|
|
|
|
// This method is a no-op if a table with the provided name doesn't exist.
|
|
|
|
//
|
|
|
|
// Be aware that this method is vulnerable to SQL injection and the
|
|
|
|
// "tableName" argument must come only from trusted input!
|
2022-07-06 23:19:05 +02:00
|
|
|
func (dao *Dao) DeleteTable(tableName string) error {
|
2023-02-18 19:33:42 +02:00
|
|
|
_, err := dao.DB().NewQuery(fmt.Sprintf(
|
|
|
|
"DROP TABLE IF EXISTS {{%s}}",
|
|
|
|
tableName,
|
|
|
|
)).Execute()
|
2022-07-06 23:19:05 +02:00
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
2022-11-13 00:38:18 +02:00
|
|
|
|
|
|
|
// Vacuum executes VACUUM on the current dao.DB() instance in order to
|
|
|
|
// reclaim unused db disk space.
|
|
|
|
func (dao *Dao) Vacuum() error {
|
|
|
|
_, err := dao.DB().NewQuery("VACUUM").Execute()
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|