MySQL adapter
The mysql
adapter for MySQL wraps the github.com/go-sql-driver/mysql
driver written by Julien Schmidt.
Here you’ll learn about the particularities of the MySQL adapter. Before starting to read this detailed information, it is advisable that you take a look at the getting started page so you become acquainted with the basics of
upper/db
and you can grasp concepts better.
Installation
Use go get
to download and install the adapter:
go get github.com/upper/db/v4/adapter/mysql
Setup
Database Session
Import the mysql
package into your application:
// main.go
package main
import (
"github.com/upper/db/v4/adapter/mysql"
)
Define the mysql.ConnectionURL{}
struct:
// ConnectionURL defines the DSN attributes.
type ConnectionURL struct {
User string
Password string
Host string
Database string
Options map[string]string
}
Pass the mysql.ConnectionURL
value as argument to mysql.Open()
so the
session is created.
settings = mysql.ConnectionURL{
...
}
sess, err = mysql.Open(settings)
...
The
mysql.ParseURL()
function is also provided in case you need to convert the DSN into amysql.ConnectionURL
:
// ParseURL parses a DSN into a ConnectionURL struct.
mysql.ParseURL(dsn string) (ConnectionURL, error)
Common Database Operations
Once the connection is established, you can start performing operations on the database.
Example
In the following example, a table named ‘birthday’ consisting of two columns (‘name’ and ‘born’) will be created. Before starting, the table will be searched in the database and, in the event it already exists, it will be removed. Then, three rows will be inserted into the table and checked for accuracy. To this end, the database will be queried and the matches (insertions) will be printed to standard output.
The birthday
table with the name
and born
columns is created with these
SQL statements:
--' example.sql
DROP TABLE IF EXISTS birthday;
CREATE TABLE birthday (
`name` VARCHAR(50),
`born` DATE
);
The mysql
command line tool is used to run the statements in the
upperio_tests
database.
cat example.sql | mysql -uupperio -pupperio upperio_tests
The rows are inserted into the birthday
table. The database is queried for
the insertions and is set to print them to standard output.
// example.go
package main
import (
"fmt"
"log"
"time"
"github.com/upper/db/v4/adapter/mysql" // Imports the mysql adapter.
)
var settings = mysql.ConnectionURL{
Database: `upperio_tests`, // Database name
Host: `localhost,` // Server IP or name
User: `upperio`, // Username
Password: `upperio`, // Password
}
type Birthday struct {
// The 'name' column of the 'birthday' table
// is mapped to the 'name' property.
Name string `db:"name"`
// The 'born' column of the 'birthday' table
// is mapped to the 'born' property.
Born time.Time `db:"born"`
}
func main() {
// The database connection is attempted.
sess, err := mysql.Open(settings)
if err != nil {
log.Fatalf("db.Open(): %q\n", err)
}
defer sess.Close() // Closing the session is a good practice.
// The 'birthday' table is referenced.
birthdayCollection := sess.Collection("birthday")
// Any rows that might have been added between the creation of
// the table and the execution of this function are removed.
err = birthdayCollection.Truncate()
if err != nil {
log.Fatalf("Truncate(): %q\n", err)
}
// Three rows are inserted into the 'Birthday' table.
birthdayCollection.Insert(Birthday{
Name: "Hayao Miyazaki",
Born: time.Date(1941, time.January, 5, 0, 0, 0, 0, time.Local),
})
birthdayCollection.Insert(Birthday{
Name: "Nobuo Uematsu",
Born: time.Date(1959, time.March, 21, 0, 0, 0, 0, time.Local),
})
birthdayCollection.Insert(Birthday{
Name: "Hironobu Sakaguchi",
Born: time.Date(1962, time.November, 25, 0, 0, 0, 0, time.Local),
})
// The database is queried for the rows inserted.
var res db.Result
res = birthdayCollection.Find()
// The 'birthdays' variable is filled with the results found.
var birthdays []Birthday
err = res.All(&birthdays)
if err != nil {
log.Fatalf("res.All(): %q\n", err)
}
// The 'birthdays' variable is printed to stdout.
for _, birthday := range birthdays {
fmt.Printf("%s was born in %s.\n",
birthday.Name,
birthday.Born.Format("January 2, 2006"),
)
}
}
The Go file is compiled and executed using go run
:
go run example.go
The output consists of three rows including names and birthdates:
Hayao Miyazaki was born in January 5, 1941.
Nobuo Uematsu was born in March 21, 1959.
Hironobu Sakaguchi was born in November 25, 1962.
Specifications
JSON Types
You can save and retrieve data when using JSON
types. If you want to try
this out, make sure the column type is json
and the field type is
mysql.JSON
:
import (
...
"github.com/upper/db/v4/adapter/mysql"
...
)
type Person struct {
...
Properties mysql.JSON `db:"properties"`
Meta mysql.JSON `db:"meta"`
}
JSON types area supported on MySQL 8.0+.
SQL Builder
You can use the SQL builder for any complex SQL query:
q := b.SQL().Select(
"p.id",
"p.title AS publication_title",
"a.name AS artist_name",
).From("artists AS a", "publication AS p").
Where("a.id = p.author_id")
var publications []Publication
if err = q.All(&publications); err != nil {
log.Fatal(err)
}
Auto-incremental Keys
If you want to generate a unique number automatically whenever a new record is
inserted, you can use auto-incremental keys. In this case, the column must be
defined as NOT NULL AUTO_INCREMENT
.
In order for the ID to be returned by
db.Collection.Insert()
, theAUTO_INCREMENT
column must be set asPRIMARY KEY
too.
CREATE TABLE foo(
id BIGINT(20) UNSIGNED NOT NULL AUTO_INCREMENT, PRIMARY KEY(id),
title VARCHAR(255)
);
Remember to use omitempty
to specify that the ID field should be ignored if
it has an empty value:
type Foo struct {
ID int64 `db:"id,omitempty"`
Title string `db:"title"`
}
Otherwise, an error will be returned.
Escape Sequences
There might be characters that cannot be typed in the context you're working,
or else would have an undesired interpretation. Through db.Func
you can
encode the syntactic entities that cannot be directly represented by the
alphabet:
res = sess.Find().Select(db.Func("DISTINCT", "name"))
On the other hand, you can use the db.Raw
function so a given value is taken
literally:
res = sess.Find().Select(db.Raw("DISTINCT(name)"))
db.Raw
can also be used as a condition argument, similarly todb.Cond
.
Take the tour
Get the real upper/db
experience, take the tour.