Skip to content
This repository has been archived by the owner on Nov 16, 2022. It is now read-only.

Latest commit

 

History

History
490 lines (393 loc) · 16.8 KB

README.md

File metadata and controls

490 lines (393 loc) · 16.8 KB

Go Build Go Report Card codecov

Lingo

Lingo at it's most basic is a type-safe query builder and bundled execution framework that ensures your SQL queries and commands are accurate and safe, and checks all the Go SQL errors for you.

The idea is not to write an ORM but instead to help (1) build dynamic queries, and (2) quickly write code to execute those queries against a sql.DB. The frameworks for the two concepts can be used together, separately, or you can use either or. It is up to you!

With Lingos' Query Building, you can interleave business logic with your query building. Need to filter a SELECT query by different columns based on different request parameters? Lingo lets you do that easily, no manual string concatenation / []interface{} appending required!

Lingo can also be used as a command line tool which uses code generation to create type safe table representations of each table by querying your actual database schema. These generated structs, when used with the various Lingo packages, ensure that you create valid SQL queries. Those same structs can be created manually, if you don't feel like running a code generator in your build process.

With Lingos' Execution Framework, an execute package wraps a native sql.DB, and provides simple methods to query a single row, multiple rows, or executing commands. The execute types accept interfaces of the sql package types, facilitating your custom transactional logic or frameworks. The Query Building and Execution Frameworks can be used completely separately.

Lingo was inspired by Querydsl and jOOQ.

Table of Contents

Setup

Run go get github.com/weworksandbox/lingo in the project root. Follow the instructions for Generate Command.

Database Type Support

Currently, only MySQL table structure parsing is supported. Note that you can build your own table files by hand using the existing interfaces which can be used with queries.

Generating Table Files

In order to use the library in conjunction with a given database schema, you must upgrade your schema to its latest version, and then run the lingo generate command with the appropriate configuration.

You can also manually create the required files that get generated.

Generate Command

You can include this lingo generate command as either a go:generate or a bash command.

Generate entity table and columns from an existing database schema

Usage:
  lingo generate [flags]

Flags:
  -d, --dir string       directory where generated file structure should go (default "./db")
      --driver string    driver name used to initialize the SQL driver (default "mysql")
      --dsn string       data source connection string
  -h, --help             help for generate
  -s, --schema strings   schema name to generate for

Global Flags:
      --config string   configuration file

Config File

A configuration file can be used to setup each CLI argument. Each config file key is the same as each CLI command argument.

dir: "db/mysql"
schema:
  - "information_schema"
driver_name: "mysql"
dsn: "root:P@ssw0rd@/?maxAllowedPacket=0&parseTime=true"

Generated Output

Every schema or table is generated with an upper case Q for types, and a lower case q for packages.

Each schema that the generator is ran for will create a schema package (q[schemaname]) and its corresponding tables under that package (q[tablename]).

In each table's q package, two files should exist. One of them is named exported.go and the other is table.go.

Exported.go

The file exported.go is a purpose built package shortcut. It helps in quick query building without needing to hold on to an instance of the table itself.

For example:

package inventory

import (
    "github.com/weworksandbox/lingo/internal/test/schema/qsakila/qinventory"
    "github.com/weworksandbox/lingo"
    "github.com/weworksandbox/lingo/expr"
    "github.com/weworksandbox/lingo/query"
    "github.com/weworksandbox/lingo/sql"
)

func CountForStore(d lingo.Dialect, storeID int16) (sql.Data, error) {
	return query.Select(expr.Count(qinventory.InventoryId())).
        From(qinventory.T()).
		Where(qinventory.StoreId().Eq(storeID)).
        ToSQL(d)
}
Table.go

The corresponding table.go is the main reference code to each table. It contains methods to give an alias to this table instance (useful during joins), and each column as its own method. Each column has its own set of methods which help build out your query. Some of the column methods are specific to the type of column, while others are generic across all column types.

Queries

The 4 basic query types exist in the library; Select, Insert, Update, Delete. Each query is in the package pkg/core/query. :warning: PLEASE NOTE!!! None of the queries below make logical sense. They are just used to explain how to use the Lingo API to get the SQL you want.

Column Type Safety

All of the column types that are generated are configurable via the config file. The column types are only used during query building. You do not need to have an exact match of DB type to a Type Safe Column type.

Out of the box, the following Type Safe columns are supported:

  • BinaryPath
  • BoolPath
  • EntityPath
  • IntPath
  • Int64Path
  • JSONPath (small wrapper of StringPath with JSON methods)
  • StringPath
  • TimePath
  • ColumnPath (a generic string defined column)

Dialect Column Type Mappings

When generating, each parser can specify its mappings.

The default MySQL mappings can be found here: internal/parse/mysql.go

Generating SQL to use with Go's SQL Package

The examples above are useless until you use the ToSQL() method of each Query type. ToSQL() requires a Query Dialect to be passed in. Doing so will either (1) generate a SQL query string and arguments or (2) return an error because your syntax or construction of the SQL query is incorrect.

Using Generated SQL

When ToSQL() is called and no error occurs, you get a sql.Data object back. This has two methods which are useful for querying a database. The first is String() string which returns a SQL string with ? for parameters. The second is an array of empty interface (interface{}) types.

Execution Example

The following is an example showing how to create a Lingo Query, create a DB connection, prepare the statement with the Lingo SQL generated, and then pass the arguments of your Query into QueryRow.

func getCount() int {
    var d lingo.Dialect = dialect.MySQL{}

    gT := qcharactersets.As("gT")
    var getCountByNameAndDescQuery = Select(Count(Star())).From(gT).Where(gT.DefaultCollateName().IsNull().And(gT.Description().Eq("uuidByte")))
    querySql, err1 := getCountByNameAndDescQuery.ToSQL(d)
    Expect(err1).ShouldNot(HaveOccurred())

    db, dbErr := sql.Open("mysql", "DSN")
    Expect(dbErr).ShouldNot(HaveOccurred())

    ctx, cancelFunc := context.WithTimeout(context.Background(), 200*time.Millisecond)
    stmt, prepareErr := db.PrepareContext(ctx, querySql.String())
    cancelFunc()
    Expect(prepareErr).ShouldNot(HaveOccurred())

    ctx, cancelFunc = context.WithTimeout(context.Background(), 200*time.Millisecond)
    row := stmt.QueryRowContext(ctx, querySql.Values()...)
    cancelFunc()

    var count int
    scanErr := row.Scan(&count)
    Expect(scanErr).ShouldNot(HaveOccurred())

    return count
}

Query Dialects

Every time one attempts to serialize a query using ToSQL(lingo.Dialect) (sql.Data, error), you are required to pass in a dialect. These dialects aid in building the SQL properly for the systems the query is to run against.

Right now, only dialect.MySQL is built / supported, however, anyone can build their own dialect.

Select Examples

Select All from Table

query.SelectFrom(qcharactersets.T())
SELECT CHARACTER_SETS.CHARACTER_SET_NAME,
       CHARACTER_SETS.DEFAULT_COLLATE_NAME,
       CHARACTER_SETS.DESCRIPTION,
       CHARACTER_SETS.MAXLEN
FROM   information_schema.CHARACTER_SETS

Values: []

Select Columns from Table

query.Select(qcharactersets.Description(), qcharactersets.Maxlen()).From(qcharactersets.T())
SELECT CHARACTER_SETS.DESCRIPTION,
       CHARACTER_SETS.MAXLEN
FROM   information_schema.CHARACTER_SETS

Values: []

Select with multiple Where Clauses

cs := qcharactersets.As("cs")
q := query.SelectFrom(cs)
q = q.Where(cs.Description().Like("utf%"))
q = q.Where(cs.Maxlen().GT(10))
SELECT cs.CHARACTER_SET_NAME,
       cs.DEFAULT_COLLATE_NAME,
       cs.DESCRIPTION,
       cs.MAXLEN
FROM   information_schema.CHARACTER_SETS AS cs
WHERE  ( cs.DESCRIPTION LIKE ?
         AND cs.MAXLEN > ? )

Values: ["utf%", 10]

Select with nested Where

In this case, note the ordering of when the Or / And are called.

cs := qcharactersets.As("cs")
query.SelectFrom(cs).Where(cs.Description().IsNull().Or(cs.CharacterSetName().Like("utf%").Or(cs.Description().Eq("other"))).And(cs.Description().In("desc1", "desc2")))
SELECT cs.CHARACTER_SET_NAME,
       cs.DEFAULT_COLLATE_NAME,
       cs.DESCRIPTION,
       cs.MAXLEN
FROM   information_schema.CHARACTER_SETS AS cs
WHERE  ( ( cs.DESCRIPTION IS NULL
            OR ( cs.CHARACTER_SET_NAME LIKE ?
                  OR cs.DESCRIPTION = ? ) )
         AND cs.DESCRIPTION IN ( ?, ? ) )

Values: ["utf%", "other", "desc1", "desc2"],

Select with Subquery

cs := qcharactersets.As("cs")
subQuery := query.Select(cs.Description()).From(cs).Where(cs.Maxlen().GT(50))
query.SelectFrom(cs).Where(cs.Description().InPaths(subQuery))
SELECT cs.CHARACTER_SET_NAME,
       cs.DEFAULT_COLLATE_NAME,
       cs.DESCRIPTION,
       cs.MAXLEN
FROM   information_schema.CHARACTER_SETS AS cs
WHERE  cs.DESCRIPTION IN (SELECT cs.DESCRIPTION
                          FROM   information_schema.CHARACTER_SETS AS cs
                          WHERE  cs.MAXLEN > ?)

Values: [50]

Insert Examples

For all insert statements, you must provide the table to insert into, and then the columns (in order) to insert into, followed by the values you want to insert. If you have auto generated columns, simply do not specify them in your Columns() functions.

For the values to insert, you can do one of the following to specify values:

  1. Use ValuesConstants() passing in Go types (which will be converted to Lingo Expressions)
  2. Use Values() passing in other columns or aggregate Expressions
  3. Use Values() with the previous bullet point, but also constants using expression.NewValue() / expression.NewValues() to specify which are constants.

Insert Into with Go Types

charSets := qcharactersets.T()
query.InsertInto(charSets).Columns(charSets.CharacterSetName(), charSets.DefaultCollateName(), charSets.Description(), charSets.Maxlen(),
).ValuesConstants("char_set_name", "default_collate", "description", 10)
INSERT INTO information_schema.character_sets
            (character_set_name,
             default_collate_name,
             description,
             maxlen)
VALUES      (?,
             ?,
             ?,
             ?)

Values: ["char_set_name", "default_collate", "description", 10]

Insert Into with Expressions

query.InsertInto(qcharactersets.T()).Columns(qcharactersets.Maxlen()).Values(expressions.Count(expressions.Star()))
INSERT INTO information_schema.character_sets
            (maxlen)
VALUES      (Count(*))

Values: []

Insert Into with Go Types & Expressions

query.InsertInto(qcharactersets.T()).Columns(
    qcharactersets.Maxlen(), qcharactersets.CharacterSetName(),
).Values(expressions.Count(expressions.Star()), expression.NewValue("default_name"))
INSERT INTO information_schema.character_sets
            (maxlen,
             character_set_name)
VALUES      (Count(*),
             ?)

Values: ["default_name"]

Insert Into with Select

charSelect := qcharactersets.As("char_select")
selQuery := query.Select(charSelect.Maxlen(), charSelect.CharacterSetName()).From(charSelect).Where(charSelect.Maxlen().GT(5))

query.InsertInto(qcharactersets.T()).Columns(qcharactersets.Maxlen(), qcharactersets.CharacterSetName()).Select(selQuery)
INSERT INTO information_schema.character_sets
            (maxlen,
             character_set_name)
SELECT char_select.maxlen,
       char_select.character_set_name
FROM   information_schema.character_sets AS char_select
WHERE  char_select.maxlen > ?

Values: [5]

Update Examples

Update Set All Rows

query.Update(qcharactersets.T()).Set(
	qcharactersets.Description().To("new_desc"),
	qcharactersets.DefaultCollateName().To("new_collate_name"),
)
UPDATE information_schema.character_sets
SET    character_sets.description = ?,
       character_sets.default_collate_name = ?

Update Set with Where

query.Update(qcharactersets.T()).Set(
	qcharactersets.Description().To("new_desc"),
	qcharactersets.DefaultCollateName().To("new_collate_name"),
).Where(qcharactersets.CharacterSetName().Eq("identity"))
UPDATE information_schema.character_sets
SET    character_sets.description = ?,
       character_sets.default_collate_name = ?
WHERE  character_sets.character_set_name = ?

Values: ["new_desc", "new_collate_name", "identity"]

Delete Examples

Delete All Rows

query.Delete(qcharactersets.T())
DELETE FROM information_schema.CHARACTER_SETS

Values: []

Delete with Where

query.Delete(qcharactersets.T()).Where(qcharactersets.Maxlen().LTOrEq(14))
DELETE FROM information_schema.character_sets
WHERE  character_sets.maxlen <= ?

Values: [14]

Delete Left Join Where

query.Delete(qcharactersets.T()).Join(
    qcollations.T(),
    expression.LeftJoin,
    qcharactersets.CharacterSetName().EqPath(qcollations.CharacterSetName()),
).Where(qcharactersets.Maxlen().LTOrEq(14))
DELETE
FROM      information_schema.character_sets
LEFT JOIN information_schema.collations
ON        character_sets.character_set_name = collations.character_set_name
WHERE     character_sets.maxlen <= ?

Values: [14]

Contributors

Adhering to the code of conduct and license, all help is welcome.

Developer Setup

Dependencies

Build lingo locally by running mage build in the root directory of this project.

Run mage -v in the root directory to show commands available to run like tests, linters, databases etc.