245 lines
4.9 KiB
Go
245 lines
4.9 KiB
Go
|
package mysql
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"net"
|
||
|
"net/url"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
// From https://github.com/go-sql-driver/mysql/blob/master/utils.go
|
||
|
var (
|
||
|
errInvalidDSNUnescaped = errors.New("Invalid DSN: Did you forget to escape a param value?")
|
||
|
errInvalidDSNAddr = errors.New("Invalid DSN: Network Address not terminated (missing closing brace)")
|
||
|
errInvalidDSNNoSlash = errors.New("Invalid DSN: Missing the slash separating the database name")
|
||
|
)
|
||
|
|
||
|
// From https://github.com/go-sql-driver/mysql/blob/master/utils.go
|
||
|
type config struct {
|
||
|
user string
|
||
|
passwd string
|
||
|
net string
|
||
|
addr string
|
||
|
dbname string
|
||
|
params map[string]string
|
||
|
}
|
||
|
|
||
|
// ConnectionURL implements a MySQL connection struct.
|
||
|
type ConnectionURL struct {
|
||
|
User string
|
||
|
Password string
|
||
|
Database string
|
||
|
Host string
|
||
|
Socket string
|
||
|
Options map[string]string
|
||
|
}
|
||
|
|
||
|
func (c ConnectionURL) String() (s string) {
|
||
|
|
||
|
if c.Database == "" {
|
||
|
return ""
|
||
|
}
|
||
|
|
||
|
// Adding username.
|
||
|
if c.User != "" {
|
||
|
s = s + c.User
|
||
|
// Adding password.
|
||
|
if c.Password != "" {
|
||
|
s = s + ":" + c.Password
|
||
|
}
|
||
|
s = s + "@"
|
||
|
}
|
||
|
|
||
|
// Adding protocol and address
|
||
|
if c.Socket != "" {
|
||
|
s = s + fmt.Sprintf("unix(%s)", c.Socket)
|
||
|
} else if c.Host != "" {
|
||
|
host, port, err := net.SplitHostPort(c.Host)
|
||
|
if err != nil {
|
||
|
host = c.Host
|
||
|
port = "3306"
|
||
|
}
|
||
|
s = s + fmt.Sprintf("tcp(%s:%s)", host, port)
|
||
|
}
|
||
|
|
||
|
// Adding database
|
||
|
s = s + "/" + c.Database
|
||
|
|
||
|
// Do we have any options?
|
||
|
if c.Options == nil {
|
||
|
c.Options = map[string]string{}
|
||
|
}
|
||
|
|
||
|
// Default options.
|
||
|
if _, ok := c.Options["charset"]; !ok {
|
||
|
c.Options["charset"] = "utf8"
|
||
|
}
|
||
|
|
||
|
if _, ok := c.Options["parseTime"]; !ok {
|
||
|
c.Options["parseTime"] = "true"
|
||
|
}
|
||
|
|
||
|
// Converting options into URL values.
|
||
|
vv := url.Values{}
|
||
|
|
||
|
for k, v := range c.Options {
|
||
|
vv.Set(k, v)
|
||
|
}
|
||
|
|
||
|
// Inserting options.
|
||
|
if p := vv.Encode(); p != "" {
|
||
|
s = s + "?" + p
|
||
|
}
|
||
|
|
||
|
return s
|
||
|
}
|
||
|
|
||
|
// ParseURL parses s into a ConnectionURL struct.
|
||
|
func ParseURL(s string) (conn ConnectionURL, err error) {
|
||
|
var cfg *config
|
||
|
|
||
|
if cfg, err = parseDSN(s); err != nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
conn.User = cfg.user
|
||
|
conn.Password = cfg.passwd
|
||
|
|
||
|
if cfg.net == "unix" {
|
||
|
conn.Socket = cfg.addr
|
||
|
} else if cfg.net == "tcp" {
|
||
|
conn.Host = cfg.addr
|
||
|
}
|
||
|
|
||
|
conn.Database = cfg.dbname
|
||
|
|
||
|
conn.Options = map[string]string{}
|
||
|
|
||
|
for k, v := range cfg.params {
|
||
|
conn.Options[k] = v
|
||
|
}
|
||
|
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// from https://github.com/go-sql-driver/mysql/blob/master/utils.go
|
||
|
// parseDSN parses the DSN string to a config
|
||
|
func parseDSN(dsn string) (cfg *config, err error) {
|
||
|
// New config with some default values
|
||
|
cfg = &config{}
|
||
|
|
||
|
// TODO: use strings.IndexByte when we can depend on Go 1.2
|
||
|
|
||
|
// [user[:password]@][net[(addr)]]/dbname[?param1=value1¶mN=valueN]
|
||
|
// Find the last '/' (since the password or the net addr might contain a '/')
|
||
|
foundSlash := false
|
||
|
for i := len(dsn) - 1; i >= 0; i-- {
|
||
|
if dsn[i] == '/' {
|
||
|
foundSlash = true
|
||
|
var j, k int
|
||
|
|
||
|
// left part is empty if i <= 0
|
||
|
if i > 0 {
|
||
|
// [username[:password]@][protocol[(address)]]
|
||
|
// Find the last '@' in dsn[:i]
|
||
|
for j = i; j >= 0; j-- {
|
||
|
if dsn[j] == '@' {
|
||
|
// username[:password]
|
||
|
// Find the first ':' in dsn[:j]
|
||
|
for k = 0; k < j; k++ {
|
||
|
if dsn[k] == ':' {
|
||
|
cfg.passwd = dsn[k+1 : j]
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
cfg.user = dsn[:k]
|
||
|
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// [protocol[(address)]]
|
||
|
// Find the first '(' in dsn[j+1:i]
|
||
|
for k = j + 1; k < i; k++ {
|
||
|
if dsn[k] == '(' {
|
||
|
// dsn[i-1] must be == ')' if an address is specified
|
||
|
if dsn[i-1] != ')' {
|
||
|
if strings.ContainsRune(dsn[k+1:i], ')') {
|
||
|
return nil, errInvalidDSNUnescaped
|
||
|
}
|
||
|
return nil, errInvalidDSNAddr
|
||
|
}
|
||
|
cfg.addr = dsn[k+1 : i-1]
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
cfg.net = dsn[j+1 : k]
|
||
|
}
|
||
|
|
||
|
// dbname[?param1=value1&...¶mN=valueN]
|
||
|
// Find the first '?' in dsn[i+1:]
|
||
|
for j = i + 1; j < len(dsn); j++ {
|
||
|
if dsn[j] == '?' {
|
||
|
if err = parseDSNParams(cfg, dsn[j+1:]); err != nil {
|
||
|
return
|
||
|
}
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
cfg.dbname = dsn[i+1 : j]
|
||
|
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if !foundSlash && len(dsn) > 0 {
|
||
|
return nil, errInvalidDSNNoSlash
|
||
|
}
|
||
|
|
||
|
// Set default network if empty
|
||
|
if cfg.net == "" {
|
||
|
cfg.net = "tcp"
|
||
|
}
|
||
|
|
||
|
// Set default address if empty
|
||
|
if cfg.addr == "" {
|
||
|
switch cfg.net {
|
||
|
case "tcp":
|
||
|
cfg.addr = "127.0.0.1:3306"
|
||
|
case "unix":
|
||
|
cfg.addr = "/tmp/mysql.sock"
|
||
|
default:
|
||
|
return nil, errors.New("Default addr for network '" + cfg.net + "' unknown")
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// From https://github.com/go-sql-driver/mysql/blob/master/utils.go
|
||
|
// parseDSNParams parses the DSN "query string"
|
||
|
// Values must be url.QueryEscape'ed
|
||
|
func parseDSNParams(cfg *config, params string) (err error) {
|
||
|
for _, v := range strings.Split(params, "&") {
|
||
|
param := strings.SplitN(v, "=", 2)
|
||
|
if len(param) != 2 {
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
value := param[1]
|
||
|
|
||
|
// lazy init
|
||
|
if cfg.params == nil {
|
||
|
cfg.params = make(map[string]string)
|
||
|
}
|
||
|
|
||
|
if cfg.params[param[0]], err = url.QueryUnescape(value); err != nil {
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return
|
||
|
}
|