Documentation ¶
Overview ¶
Package bsq helps to Build Structured Queries for SQL in a maintainable manner. Unlike many query builders out there it does not strive to formalize SQL as Go DSL. Its primary focus it code maintainability when working with SQL in your code. – Why do we need something different from the “usual” approach?
I.e. consider the problem of keeping the Go code in line with the DB model to be the primary maintenance task: It must be easy to find all parts of the code that have to change when the DB model changes. Often, query builders present an API that let's you do something like this:
sql := Select("name", "age").From("users").Where(EqArg("id"))
And this might yield the expected string:
SELECT name, age FROM users WHERE id=?
While this may help if you are not familiar with SQL, it doesn't do much. What if the users table gets a column renamed, removed or even gets a complely new column? How do you find all the relevant Go code that relate to the users table? At best a clever tool can search for all uses of the From() method that gets the string "users" as an argument. Without such tool you have to try a full text search. Package bsq takes a different approach. This approach starts with a minimal replica of the DB model made of Go objects:
var tUsers = struct { Table ID, Name, Age Column }{ Table: Table{TableName: "users"}, }
The extra effort for this replica helps much with the maintainability problem. Consider this conceptual example (that omits some syntactic sugar which actually would improve the code):
query, _ := NewQuery(dialect, "SELECT ", &tUsers.Name, ", ", &tUsers.Age, " FROM ", &tUsers, " WHERE ", &tUsers.ID, '=', P("id"), ) fmt.Println(query.String())
While this does not help with SQL syntax, one now can simply use standard developer tools to locate all references to the 'tUsers' variable. And renaming a column can be done at one single location where the Name column is declared in the replica.
Package bsq builds on this concept to simplify the creation of SQL statements. E.g. to get the example SELECT statement one would actually write with bsq:
q, _ := NewQuery(dialect, Select{ Columns: ColsLs(&tUsers, &tUsers.ID), // all from tUsers without ID Key: Cols{&tUsers.ID}, }) sql := q.String()
Defining a DB Schema in Go ¶
TODO!
See also: dbtest.VerifyTables, dbtest.VerifyTable
Writing Queries ¶
TODO!
Index ¶
- func CamelToSnake(name string) string
- func ColP(sql ...any) any
- func ColsOf(t TableDef, without ...*Column) (cols []any)
- func DefaultNameMap() func(string) string
- func EachColumn(def TableDef, do func(*Column, *reflect.StructField) error) error
- func EachTable(do func(TableDef) error) error
- func FullJoinEq(from, to ColumnRef) any
- func IndexedVars(fmt string) func() BindVars
- func InitSchema(schema string, opts *InitOpts, tables ...TableDef) error
- func JoinEq(from, to ColumnRef) any
- func LeftJoinEq(from, to ColumnRef) any
- func NamedVars(fmt string) func() BindVars
- func Out(sql ...any) any
- func P(key any) any
- func PEq(col ColumnRef) any
- func PGe(col ColumnRef) any
- func PGt(col ColumnRef) any
- func PLe(col ColumnRef) any
- func PLt(col ColumnRef) any
- func PNe(col ColumnRef) any
- func PositionalVars(sym string) func() BindVars
- func QueryCreate(db sqlize.Querier, q *Query, args ...any) (id int64, err error)
- func RightJoinEq(from, to ColumnRef) any
- type BindVars
- type Column
- type ColumnRef
- type Columns
- type CreateDialect
- type Delete
- type Dialect
- type DialectCtx
- type ExplicitMap
- type IndexBindVars
- type InitOpts
- type Insert
- type L
- type NameBindVars
- type PositionBindVars
- type Prunable
- type Queries
- type Query
- func DefCreateQuery(d Dialect, idColumn *Column, cols ...*Column) (*Query, error)
- func DefQuery(defn ...any) *Query
- func DefUpsertQuery(d Dialect, ups Upsert) (*Query, error)
- func NewCreateQuery(d Dialect, idColumn *Column, cols ...*Column) (*Query, error)
- func NewQuery(d Dialect, defn ...any) (*Query, error)
- func NewQueryPrep(db *sql.DB, d Dialect, defn ...any) (*Query, error)
- func NewQueryPrepContext(ctx context.Context, db *sql.DB, d Dialect, defn ...any) (*Query, error)
- func NewUpsertQuery(d Dialect, ups Upsert) (*Query, error)
- func (q *Query) Dialect() Dialect
- func (q *Query) Exec(db sqlize.Querier, args ...any) (sql.Result, error)
- func (q *Query) ExecContext(ctx context.Context, db sqlize.Querier, args ...any) (sql.Result, error)
- func (q *Query) Init(d Dialect) error
- func (q *Query) OutCols() []*Column
- func (q *Query) ParamCols() []*Column
- func (q *Query) Prepare(db *sql.DB) (err error)
- func (q *Query) PrepareContext(ctx context.Context, db *sql.DB) (err error)
- func (q *Query) Prune(p Prunable)
- func (q *Query) Query(db sqlize.Querier, args ...any) (*sql.Rows, error)
- func (q *Query) QueryContext(ctx context.Context, db sqlize.Querier, args ...any) (*sql.Rows, error)
- func (q *Query) QueryRow(db sqlize.Querier, args ...any) *sql.Row
- func (q *Query) QueryRowContext(ctx context.Context, db sqlize.Querier, args ...any) *sql.Row
- func (q *Query) Stmt() *sql.Stmt
- func (q *Query) String() string
- type Select
- type Table
- type TableDef
- type Update
- type Upsert
- type UpsertDialect
- type WriteCtx
Examples ¶
Constants ¶
This section is empty.
Variables ¶
This section is empty.
Functions ¶
func CamelToSnake ¶ added in v0.7.2
func DefaultNameMap ¶ added in v0.7.2
func EachColumn ¶ added in v0.7.2
Example ¶
type Entity struct { Table ID Column } tbl := struct { Entity Name Column }{ Entity: Entity{ Table: Table{Name: "yet_another_table"}, ID: Column{Name: "id"}, }, Name: Column{Name: "name"}, } err := EachColumn(&tbl, func(c *Column, f *reflect.StructField) error { fmt.Println(c.Name, "->", f.Name) return nil }) fmt.Println(err)
Output: id -> ID name -> Name <nil>
func FullJoinEq ¶ added in v0.7.2
func IndexedVars ¶ added in v0.7.2
func InitSchema ¶ added in v0.7.2
InitSchema adds the tables to the list of known tables. It does not check for duplicates because there may be reasonable applications that hold table definitions for equally named tables from different databases. The schema name of each table is set according to opts.ForceSchema and the default alias of each table is checked according to opts.MustAlias.
func LeftJoinEq ¶ added in v0.7.2
func P ¶
P adds a parameter to a query definition. The parameter key must either be a string or a ColumnRef to identify the parameter. The key lets the query generator recognize the same parameter when used several times in a query.
Example ¶
example := func(bv func() BindVars, p, q any) { qry, _ := NewQuery(DefaultSQL(bv, nil), p, AND, q) fmt.Println(len(qry.ParamCols()), "parameters:", qry.String()) } example(PositionalVars("?"), P("p1"), P("p1")) example(IndexedVars("$%d"), P("p1"), P("p1")) example(IndexedVars("$%d"), P("p1"), P("p2")) example(NamedVars(":%s"), P("p1"), P("p1")) example(NamedVars(":%s"), P("p1"), P("p2"))
Output: 2 parameters: ? AND ? 1 parameters: $1 AND $1 2 parameters: $1 AND $2 1 parameters: :p1 AND :p1 2 parameters: :p1 AND :p2
func PEq ¶
PEq adds 'column-name'='bind-var' to the query
Example ¶
q, _ := NewQuery(testDialect, PEq(&testTable.Name)) fmt.Println(q.String())
Output: Name=?
func PGe ¶ added in v0.7.2
PGe adds 'column-name'>='bind-var' to the query
Example ¶
q, _ := NewQuery(testDialect, PGe(&testTable.Name)) fmt.Println(q.String())
Output: Name>=?
func PGt ¶ added in v0.7.2
PGt adds 'column-name'>'bind-var' to the query
Example ¶
q, _ := NewQuery(testDialect, PGt(&testTable.Name)) fmt.Println(q.String())
Output: Name>?
func PLe ¶ added in v0.7.2
PLe adds 'column-name'<='bind-var' to the query
Example ¶
q, _ := NewQuery(testDialect, PLe(&testTable.Name)) fmt.Println(q.String())
Output: Name<=?
func PLt ¶ added in v0.7.2
PLt adds 'column-name'<'bind-var' to the query
Example ¶
q, _ := NewQuery(testDialect, PLt(&testTable.Name)) fmt.Println(q.String())
Output: Name<?
func PNe ¶ added in v0.7.2
PNe adds 'column-name'<>'bind-var' to the query
Example ¶
q, _ := NewQuery(testDialect, PNe(&testTable.Name)) fmt.Println(q.String())
Output: Name<>?
func PositionalVars ¶ added in v0.7.2
func QueryCreate ¶ added in v0.2.0
TODO what about prepared queries
func RightJoinEq ¶ added in v0.7.2
Types ¶
type Column ¶
type Column struct { Name string // contains filtered or unexported fields }
type Columns ¶ added in v0.7.2
type Columns []*Column
type CreateDialect ¶ added in v0.7.2
type Delete ¶
type Delete struct {
Key Columns
}
Example ¶
del := Delete{Key: Columns{&testTable.ID}} // Delete by key q := must.Ret(NewQuery(testDialect, del)) fmt.Println(q.String())
Output: DELETE FROM thing WHERE ID=?
type Dialect ¶
type Dialect interface { NewContext() DialectCtx Quote(name string) string Parameter(ctx DialectCtx, key string, col *Column) (string, error) }
type DialectCtx ¶ added in v0.7.2
type ExplicitMap ¶ added in v0.7.2
type ExplicitMap struct {
// contains filtered or unexported fields
}
func NewExplicitMap ¶ added in v0.7.2
func NewExplicitMap(mapping map[string]string, fallback func(string) string) *ExplicitMap
func (ExplicitMap) Map ¶ added in v0.7.2
func (m ExplicitMap) Map(name string) string
func (ExplicitMap) With ¶ added in v0.7.2
func (m ExplicitMap) With(mapping map[string]string) ExplicitMap
type IndexBindVars ¶ added in v0.7.2
type IndexBindVars struct {
// contains filtered or unexported fields
}
func NewIndexBindVars ¶ added in v0.7.2
func NewIndexBindVars(format string) *IndexBindVars
func (*IndexBindVars) Param ¶ added in v0.7.2
func (bv *IndexBindVars) Param(key string, c *Column) (string, error)
func (*IndexBindVars) ParamColumns ¶ added in v0.7.2
func (bv *IndexBindVars) ParamColumns() []*Column
type InitOpts ¶
type InitOpts struct { // If true InitSchema will set the passed schema on all tables. Otherwise, // the schema will be set only if the table does not yet have a schema name. ForceSchema bool // If true InitSchema will return an error if a table does not have a // default alias. MustAlias bool // Used to set column names from struct field names. This is only used on // columns that do not yet have a name set. If ColNameMap is nil, the struct // filed name will be used as is. ColNameMap func(fieldname string) (columname string) }
type Insert ¶
Example ¶
ins := Insert{Table: &testTable} // Isert all columns of table q := must.Ret(NewQuery(testDialect, ins)) fmt.Println(q.String())
Output: INSERT INTO thing (ID, Name) VALUES (?, ?)
type NameBindVars ¶ added in v0.7.2
type NameBindVars struct {
// contains filtered or unexported fields
}
func NewNamedBindVars ¶ added in v0.7.2
func NewNamedBindVars(format string) *NameBindVars
func (*NameBindVars) Param ¶ added in v0.7.2
func (bv *NameBindVars) Param(key string, c *Column) (string, error)
func (*NameBindVars) ParamColumns ¶ added in v0.7.2
func (bv *NameBindVars) ParamColumns() []*Column
type PositionBindVars ¶ added in v0.7.2
type PositionBindVars struct { Symbol string // contains filtered or unexported fields }
func (*PositionBindVars) Param ¶ added in v0.7.2
func (bv *PositionBindVars) Param(_ string, c *Column) (string, error)
func (*PositionBindVars) ParamColumns ¶ added in v0.7.2
func (bv *PositionBindVars) ParamColumns() []*Column
type Queries ¶ added in v0.7.2
type Queries struct {
// contains filtered or unexported fields
}
Queries is used to collect a set of queries and later apply an operation to all queries in the set - typically to initialise and/or prepare them. A zero value is valid to start collecting queries.
func NewQueries ¶ added in v0.7.2
NewQueries is used to hierachically create a collection of queries. The sub-collections are passed as arguemnts. Then the new Queries object can be used to add more queries.
func (*Queries) Add ¶ added in v0.7.2
Add adds a queriy to the collection. Add is safe for concurrent use, also with ForEach.
type Query ¶
type Query struct {
// contains filtered or unexported fields
}
Example ¶
testTable := struct { Table ID, Name Column }{Table: Table{Name: "test", DefaultAlias: "t"}} InitSchema("", nil, &testTable) testAlias := testTable.As("t2") q, err := NewQuery(testDialect, SELECT, Out(&testTable.ID, testTable.Name.In(testAlias)), FROM, Ls(&testTable, testAlias), WHERE, P("name"), IS_NOT_NULL, AND, PGe(&testTable.ID), AND, PLt(&testTable.ID), ) if err != nil { fmt.Println(err) return } fmt.Println(q.String()) fmt.Println(" ^ o1 ^o2 ^ p1 ^ p2 ^ p3") fmt.Print("Param Columns") for i, c := range q.pCols { if c == nil { fmt.Printf(" %d:-", i+1) } else { fmt.Printf(" %d:%s", i+1, c.QName()) } } fmt.Print("\nOutput") for i, c := range q.oCols { if c == nil { fmt.Printf(" %d:-", i+1) } else { fmt.Printf(" %d:%s", i+1, c.QName()) } }
Output: SELECT t.ID, t2.Name FROM test t, test t2 WHERE ? IS NOT NULL AND t.ID>=? AND t.ID<? ^ o1 ^o2 ^ p1 ^ p2 ^ p3 Param Columns 1:- 2:test.ID 3:test.ID Output 1:test.ID 2:test.Name
func DefCreateQuery ¶ added in v0.7.2
func NewCreateQuery ¶ added in v0.7.2
func NewQuery ¶ added in v0.2.0
NewQuery first checks once if defn has only one element with a Defn() []any method. If so, it uses the result from Defn() to replace defn. Then it creates and Inits a new Query with Dialect=d and Defn=defn.
func NewQueryPrep ¶ added in v0.7.2
func NewQueryPrepContext ¶ added in v0.7.2
func (*Query) ExecContext ¶ added in v0.2.0
func (*Query) Init ¶ added in v0.7.2
Init initializes the Query according to its definition. If d is not nil it will initialize q.dialect. Trying to override an already set dialect results in an error. Init must not be called before InitSchema for the tables used in the query. One may use Queries to initialize a set of global Query variables.
func (*Query) PrepareContext ¶ added in v0.2.0
func (*Query) QueryContext ¶ added in v0.2.0
func (*Query) QueryRowContext ¶ added in v0.2.0
type Select ¶
type Select tabColsKey
Example ¶
sel := Select{Table: &testTable} // Select * from table q := must.Ret(NewQuery(testDialect, sel)) fmt.Println(q.String()) sel = Select{Columns: Cols(&testTable)} // Select all columns from table q = must.Ret(NewQuery(testDialect, sel)) fmt.Println(q.String()) sel = Select{ // Select column(s) by key Columns: nil, Key: Columns{&testTable.ID}, } q = must.Ret(NewQuery(testDialect, sel)) fmt.Println(q.String()) sel = Select{ // Select column(s) by key Columns: Columns{&testTable.Name}, Key: Columns{&testTable.ID}, } q = must.Ret(NewQuery(testDialect, sel)) fmt.Println(q.String())
Output: SELECT * FROM thing SELECT ID, Name FROM thing SELECT Name FROM thing WHERE ID=? SELECT Name FROM thing WHERE ID=?
type Table ¶
type Table struct { // The schema name of the declared table. It can also be set with // InitSchema. Schema string // The table name. Name string // A default alias for SQL queries. Specific aliases can be created with the // Alias method. DefaultAlias string // contains filtered or unexported fields }
Table is used to declare the table data for the DB model replica. Replica are used to define SQL queries.
To see how to use it, read example of EachColumn and InitSchema.
type Update ¶
type Update tabColsKey
Example ¶
upd := Update{Table: &testTable} // Update all columns q := must.Ret(NewQuery(DefaultSQL(IndexedVars("$%d"), nil), upd)) fmt.Println(q.String()) upd = Update{Key: Columns{&testTable.ID}} // Update all but key columns q = must.Ret(NewQuery(DefaultSQL(IndexedVars("$%d"), nil), upd)) fmt.Println(q.String())
Output: UPDATE thing SET ID=$1, Name=$2 UPDATE thing SET Name=$1 WHERE ID=$2
type UpsertDialect ¶ added in v0.7.2
type WriteCtx ¶ added in v0.7.2
type WriteCtx struct {
// contains filtered or unexported fields
}