Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/arthurkushman/buildsqlx

Go database query builder library for PostgreSQL
https://github.com/arthurkushman/buildsqlx

active-record clauses database-table ddl dml go golang golang-library postgresql query-builder sql sqlbuilder transactions

Last synced: 27 days ago
JSON representation

Go database query builder library for PostgreSQL

Awesome Lists containing this project

README

        

# buildsqlx

Go Database query builder
library [![Tweet](http://jpillora.com/github-twitter-button/img/tweet.png)](https://twitter.com/intent/tweet?text=Go%20database%20query%20builder%20library%20&url=https://github.com/arthurkushman/buildsqlx&hashtags=go,golang,sql,builder,postgresql,sql-builder,developers)

[![Mentioned in Awesome Go](https://awesome.re/mentioned-badge.svg)](https://github.com/avelino/awesome-go)
[![Go Report Card](https://goreportcard.com/badge/github.com/arthurkushman/buildsqlx)](https://goreportcard.com/report/github.com/arthurkushman/buildsqlx)
[![Build and run](https://github.com/arthurkushman/buildsqlx/workflows/Build%20and%20run/badge.svg)](https://github.com/arthurkushman/buildsqlx/actions)
[![GoDoc](https://github.com/golang/gddo/blob/c782c79e0a3c3282dacdaaebeff9e6fd99cb2919/gddo-server/assets/status.svg)](https://godoc.org/github.com/arthurkushman/buildsqlx)
[![codecov](https://codecov.io/gh/arthurkushman/buildsqlx/branch/master/graph/badge.svg)](https://codecov.io/gh/arthurkushman/buildsqlx)
[![License: MIT](https://img.shields.io/badge/License-MIT-blue.svg)](https://opensource.org/licenses/MIT)

* [Installation](#user-content-installation)
* [Selects, Ordering, Limit & Offset](#user-content-selects-ordering-limit--offset)
* [GroupBy / Having](#user-content-groupby--having)
* [Where, AndWhere, OrWhere clauses](#user-content-where-andwhere-orwhere-clauses)
* [WhereIn / WhereNotIn](#user-content-wherein--wherenotin)
* [WhereNull / WhereNotNull](#user-content-wherenull--wherenotnull)
* [Left / Right / Cross / Inner / Left Outer Joins](#user-content-left--right--cross--inner--left-outer-joins)
* [Inserts](#user-content-inserts)
* [Updates](#user-content-updates)
* [Delete](#user-content-delete)
* [Drop, Truncate, Rename](#user-content-drop-truncate-rename)
* [Increment & Decrement](#user-content-increment--decrement)
* [Union / Union All](#user-content-union--union-all)
* [Transaction mode](#user-content-transaction-mode)
* [Dump, Dd](#user-content-dump-dd)
* [Check if table exists](#user-content-check-if-table-exists)
* [Check if columns exist in a table within schema](#user-content-check-if-columns-exist-in-a-table-within-schema)
* [Retrieving A Single Row / Column From A Table](#user-content-retrieving-a-single-row--column-from-a-table)
* [WhereExists / WhereNotExists](#user-content-whereexists--wherenotexists)
* [Determining If Records Exist](#user-content-determining-if-records-exist)
* [Aggregates](#user-content-aggregates)
* [Create table](#user-content-create-table)
* [Add / Modify / Drop columns](#user-content-add--modify--drop-columns)
* [Chunking Results](#user-content-chunking-results)
* [Pluck / PluckMap](#user-content-pluck--pluckmap)

## Installation

```bash
go get -u github.com/arthurkushman/buildsqlx
```

## Selects, Ordering, Limit & Offset

You may not always want to select all columns from a database table. Using the select method, you can specify a custom
select clause for the query:

```go
package yourpackage

import (
"database/sql"

"github.com/arthurkushman/buildsqlx"
_ "github.com/lib/pq"
)

var db = buildsqlx.NewDb(buildsqlx.NewConnection("postgres", "user=postgres dbname=postgres password=postgres sslmode=disable"))

func main() {
qDb := db.Table("posts").Select("title", "body")

type DataStruct struct {
Foo string
Bar string
Baz *int64
}

dataStruct := DataStruct{}
var testStructs []DataStruct
// If you already have a query builder instance and you wish to add a column to its existing select clause, you may use the addSelect method:
err := qDb.AddSelect("points").GroupBy("topic").OrderBy("points", "DESC").Limit(15).Offset(5).EachToStruct(func(rows *sql.Rows) error {
err = db.Next(rows, &dataStruct)
if err != nil {
return err
}

testStructs = append(testStructs, dataStruct)
return nil
})
}
```

### InRandomOrder

```go
err = db.Table("users").Select("name", "post", "user_id").InRandomOrder().ScanStruct(dataStruct)
```

## GroupBy / Having

The `GroupBy` and `Having` methods may be used to group the query results.
The having method's signature is similar to that of the `Where` method:

```go
err = db.table("users").GroupBy("account_id").Having("account_id", ">", 100).ScanStruct(dataStruct)
```

## Where, AndWhere, OrWhere clauses

You may use the `Where` method on a query builder instance to add where clauses to the query.
The most basic call to where requires three arguments.
The first argument is the name of the column.
The second argument is an operator, which can be any of the database's supported operators.
Finally, the third argument is the value to evaluate against the column.

```go
err = db.Table("table1").Select("foo", "bar", "baz").Where("foo", "=", cmp).AndWhere("bar", "!=", "foo").OrWhere("baz", "=", 123)..ScanStruct(dataStruct)
```

You may chain where constraints together as well as add or clauses to the query.
The `OrWhere` method accepts the same arguments as the `Where` method.

## WhereIn / WhereNotIn

The `WhereIn` method verifies that a given column's value is contained within the given slice:

```go
err = db.Table("table1").WhereIn("id", []int64{1, 2, 3}).OrWhereIn("name", []string{"John", "Paul"}).ScanStruct(dataStruct)
```

## WhereNull / WhereNotNull

The `WhereNull` method verifies that the value of the given column is `NULL`:

```go
err = db.Table("posts").WhereNull("points").OrWhereNotNull("title")..ScanStruct(dataStruct)
```

## Left / Right / Cross / Inner / Left Outer Joins

The query builder may also be used to write join statements.
To perform a basic "inner join", you may use the `InnerJoin` method on a query builder instance.
The first argument passed to the join method is the name of the table you need to join to,
while the remaining arguments specify the column constraints for the join.
You can even join to multiple tables in a single query:

```go
err = db.Table("users").Select("name", "post", "user_id").LeftJoin("posts", "users.id", "=", "posts.user_id").EachToStruct(func(rows *sql.Rows) error {
err = db.Next(rows, &dataStruct)
if err != nil {
return err
}

testStructs = append(testStructs, dataStruct)
return nil
})
```

## Inserts

The query builder also provides an `Insert` method for inserting records into the database table.
The `Insert/InsertBatch` methods accept a structure (or slice of structs) of column names and values:

```go
// insert without getting id
err = db.Table("table1").Insert(DataStruct{
Foo: "foo foo foo",
Bar: "bar bar bar",
Baz: &baz,
})

// insert returning id
id, err := db.Table("table1").InsertGetId(DataStruct{
Foo: "foo foo foo",
Bar: "bar bar bar",
Baz: &baz,
})

// batch insert
err = db.Table("table1").InsertBatch([]DataStruct{
{Foo: "foo foo foo", Bar: "bar bar bar", Baz: &baz},
{Foo: "foo foo foo foo", Bar: "bar bar bar bar", Baz: &baz},
{Foo: "foo foo foo foo foo", Bar: "bar bar bar bar bar", Baz: &baz},
})
```

## Updates

In addition to inserting records into the database,
the query builder can also update existing records using the update method.
The update method, like the insert method, accepts a slice of column and value pairs containing the columns to be
updated.
You may constrain the update query using where clauses:

```go
rows, err := db.Table("posts").Where("points", ">", 3).Update(DataStruct{
Title: "awesome",
})
```

## Delete

The query builder may also be used to delete records from the table via the delete method.
You may constrain delete statements by adding where clauses before calling the delete method:

```go
rows, err := db.Table("posts").Where("points", "=", 123).Delete()
```

## Drop, Truncate, Rename

```go
db.Drop("table_name")

db.DropIfExists("table_name")

db.Truncate("table_name")

db.Rename("table_name1", "table_name2")
```

## Increment & Decrement

The query builder also provides convenient methods for incrementing or decrementing the value of a given column.
This is a shortcut, providing a more expressive and terse interface compared to manually writing the update statement.

Both of these methods accept 2 arguments: the column to modify, a second argument to control the amount by which the
column should be incremented or decremented:

```go
db.Table("users").Increment("votes", 3)

db.Table("users").Decrement("votes", 1)
```

## Union / Union All

The query builder also provides a quick way to "union" two queries together.
For example, you may create an initial query and use the union method to union it with a second query:

```go
union := db.Table("posts").Select("title", "likes").Union()
res, err := union.Table("users").Select("name", "points").ScanStruct(dataStruct)

// or if UNION ALL is of need
// union := db.Table("posts").Select("title", "likes").UnionAll()
```

## Transaction mode

You can run arbitrary queries mixed with any code in transaction mode getting an error and as a result rollback if
something went wrong
or committed if everything is ok:

```go
err := db.InTransaction(func () (interface{}, error) {
return db.Table("users").Select("name", "post", "user_id").ScanStruct(dataStruct)
})
```

## Dump, Dd

You may use the Dd or Dump methods while building a query to dump the query bindings and SQL.
The dd method will display the debug information and then stop executing the request.
The dump method will display the debug information but allow the request to keep executing:

```go
// to print raw sql query to stdout
db.Table("table_name").Select("foo", "bar", "baz").Where("foo", "=", cmp).AndWhere("bar", "!=", "foo").Dump()

// or to print to stdout and exit a.k.a dump and die
db.Table("table_name").Select("foo", "bar", "baz").Where("foo", "=", cmp).AndWhere("bar", "!=", "foo").Dd()
```

## Check if table exists

```go
tblExists, err := db.HasTable("public", "posts")
```

## Check if columns exist in a table within schema

```go
colsExists, err := db.HasColumns("public", "posts", "title", "user_id")
```

## Retrieving A Single Row / Column From A Table

If you just need to retrieve a single row from the database table, you may use the `First` func.
This method will return a single `map[string]interface{}`:

```go
err = db.Table("posts").Select("title").OrderBy("created_at", "desc").First(dataStruct)

// usage ex: dataStruct.Title
```

If you don't even need an entire row, you may extract a single value from a record using the `Value` method.
This method will return the value of the column directly:

```go
err = db.Table("users").OrderBy("points", "desc").Value(dataStruct, "name")

// dataStruct.Name -> "Alex Shmidt"
```

To retrieve a single row by its id column value, use the `find` method:

```go
user, err := db.Table("users").Find(dataStruct, id)

// dataStruct.ID, dataStruct.Name, dataStruct.Email etc
```

## WhereExists / WhereNotExists

The whereExists method allows you to write where exists SQL clauses.
The whereExists method accepts a *DB argument,
which will receive a query builder instance allowing you to define the query that should be placed inside the "exists" clause:

```go
err = db.Table("users").Select("name").WhereExists(
db.Table("users").Select("name").Where("points", ">=", int64(12345)),
).First(dataStruct)
```

Any query that is of need to build one can place inside `WhereExists` clause/func.

## WhereBetween / WhereNotBetween

The whereBetween func verifies that a column's value is between two values:

```go
err = db.Table(UsersTable).Select("name").WhereBetween("points", 1233, 12345).ScanStruct(&testStruct)
```

The whereNotBetween func verifies that a column's value lies outside of two values:

```go
err = db.Table(UsersTable).Select("name").WhereNotBetween("points", 123, 123456).ScanStruct(&testStruct)
```

## Determining If Records Exist

Instead of using the `Count` method to determine if any records exist that match your query's constraints,
you may use the exists and doesntExist methods:

```go
exists, err := db.Table(UsersTable).Select("name").Where("points", ">=", int64(12345)).Exists()
// use an inverse DoesntExists() if needed
```

## Aggregates

The query builder also provides a variety of aggregate methods such as Count, Max, Min, Avg, and Sum.
You may call any of these methods after constructing your query:

```go
cnt, err := db.Table(UsersTable).WHere("points", ">=", 1234).Count()

avg, err := db.Table(UsersTable).Avg("points")

mx, err := db.Table(UsersTable).Max("points")

mn, err := db.Table(UsersTable).Min("points")

sum, err := db.Table(UsersTable).Sum("points")
```

## Create table

To create a new database table, use the CreateTable method.
The Schema method accepts two arguments.
The first is the name of the table, while the second is an anonymous function/closure which receives a Table struct that
may be used to define the new table:

```go
res, err := db.Schema("big_tbl", func(table *Table) error {
table.Increments("id")
table.String("title", 128).Default("The quick brown fox jumped over the lazy dog").Unique("idx_ttl")
table.SmallInt("cnt").Default(1)
table.Integer("points").NotNull()
table.BigInt("likes").Index("idx_likes")
table.Text("comment").Comment("user comment").Collation("de_DE")
table.DblPrecision("likes_to_points").Default(0.0)
table.Char("tag", 10)
table.DateTime("created_at", true)
table.DateTimeTz("updated_at", true)
table.Decimal("tax", 2, 2)
table.TsVector("body")
table.TsQuery("body_query")
table.Jsonb("settings")
table.Point("pt")
table.Polygon("poly")
table.TableComment("big table for big data")

return nil
})

// to make a foreign key constraint from another table
_, err = db.Schema("tbl_to_ref", func (table *Table) error {
table.Increments("id")
table.Integer("big_tbl_id").ForeignKey("fk_idx_big_tbl_id", "big_tbl", "id").Concurrently().IfNotExists()
// to add index on existing column just repeat stmt + index e.g.:
table.Char("tag", 10).Index("idx_tag").Include("likes", "created_at")
table.Rename("settings", "options")

return nil
})
```

## Add / Modify / Drop columns

The Table structure in the Schema's 2nd argument may be used to update existing tables. Just the way you've been created
it.
The Change method allows you to modify some existing column types to a new type or modify the column's attributes.

```go
res, err := db.Schema("tbl_name", func(table *Table) error {
table.String("title", 128).Change()

return nil
})
```

Use DropColumn method to remove any column:

```go
res, err := db.Schema("tbl_name", func(table *Table) error {
table.DropColumn("deleted_at").IfExists()
// To drop an index on the column
table.DropIndex("idx_title")

return nil
})
```

## Chunking Results

If you need to work with thousands of database records, consider using the chunk method.
This method retrieves a small chunk of the results at a time and feeds each chunk into a closure for processing.

```go
var sumOfPoints int64
dataStruct := &DataStructUser{}
err = db.Table(UsersTable).Select("name", "points").Chunk(dataStruct, 100, func(users []any) bool {
for _, v := range users {
user := v.(DataStructUser)
// your code goes here e.g.:
sumOfPoints += user.Points
}

// or you can return false here to stop running chunks
return true
})
```

## Pluck / PluckMap

If you would like to get values of a particular column(s) of a struct and place them into slice - use `Pluck` method:
```go
dataStruct := &DataStructUser{}
res, err := db.Table(UsersTable).Pluck(dataStruct)
for k, v := range res {
val := v.(DataStructUser)
fmt.Println(val.Name) // f.e.: Alex Shmidt
}

// or use a PluckMap method to aggregate key/value pairs to a map
res, err := db.Table(UsersTable).PluckMap(dataStruct, "name", "points")
for k, m := range res {
for key, value := range m {
keyVal := key.(string)
valueVal := value.(DataStructUser)
// rest of the code ...
}
}
```

PS Why use buildsqlx? Because it is simple and fast, yet versatile. The builder code-style has been inherited from greatest web-frameworks, so u can easily query anything from db.

Supporters gratitude:

JetBrains logo