package pg import ( "crypto/tls" "errors" "fmt" "net" "net/url" "runtime" "strings" "time" "github.com/go-pg/pg/internal/pool" ) // Database connection options. type Options struct { // Network type, either tcp or unix. // Default is tcp. Network string // TCP host:port or Unix socket depending on Network. Addr string // Dialer creates new network connection and has priority over // Network and Addr options. Dialer func(network, addr string) (net.Conn, error) // Hook that is called when new connection is established. OnConnect func(*Conn) error User string Password string Database string // ApplicationName is the application name. Used in logs on Pg side. // Only availaible from pg-9.0. ApplicationName string // TLS config for secure connections. TLSConfig *tls.Config // Maximum number of retries before giving up. // Default is to not retry failed queries. MaxRetries int // Whether to retry queries cancelled because of statement_timeout. RetryStatementTimeout bool // Minimum backoff between each retry. // Default is 250 milliseconds; -1 disables backoff. MinRetryBackoff time.Duration // Maximum backoff between each retry. // Default is 4 seconds; -1 disables backoff. MaxRetryBackoff time.Duration // Dial timeout for establishing new connections. // Default is 5 seconds. DialTimeout time.Duration // Timeout for socket reads. If reached, commands will fail // with a timeout instead of blocking. ReadTimeout time.Duration // Timeout for socket writes. If reached, commands will fail // with a timeout instead of blocking. WriteTimeout time.Duration // Maximum number of socket connections. // Default is 10 connections per every CPU as reported by runtime.NumCPU. PoolSize int // Minimum number of idle connections which is useful when establishing // new connection is slow. MinIdleConns int // Connection age at which client retires (closes) the connection. // It is useful with proxies like PgBouncer and HAProxy. // Default is to not close aged connections. MaxConnAge time.Duration // Time for which client waits for free connection if all // connections are busy before returning an error. // Default is 30 seconds if ReadTimeOut is not defined, otherwise, // ReadTimeout + 1 second. PoolTimeout time.Duration // Amount of time after which client closes idle connections. // Should be less than server's timeout. // Default is 5 minutes. -1 disables idle timeout check. IdleTimeout time.Duration // Frequency of idle checks made by idle connections reaper. // Default is 1 minute. -1 disables idle connections reaper, // but idle connections are still discarded by the client // if IdleTimeout is set. IdleCheckFrequency time.Duration } func (opt *Options) init() { if opt.Network == "" { opt.Network = "tcp" } if opt.Addr == "" { switch opt.Network { case "tcp": opt.Addr = "localhost:5432" case "unix": opt.Addr = "/var/run/postgresql/.s.PGSQL.5432" } } if opt.PoolSize == 0 { opt.PoolSize = 10 * runtime.NumCPU() } if opt.PoolTimeout == 0 { if opt.ReadTimeout != 0 { opt.PoolTimeout = opt.ReadTimeout + time.Second } else { opt.PoolTimeout = 30 * time.Second } } if opt.DialTimeout == 0 { opt.DialTimeout = 5 * time.Second } if opt.IdleTimeout == 0 { opt.IdleTimeout = 5 * time.Minute } if opt.IdleCheckFrequency == 0 { opt.IdleCheckFrequency = time.Minute } switch opt.MinRetryBackoff { case -1: opt.MinRetryBackoff = 0 case 0: opt.MinRetryBackoff = 250 * time.Millisecond } switch opt.MaxRetryBackoff { case -1: opt.MaxRetryBackoff = 0 case 0: opt.MaxRetryBackoff = 4 * time.Second } } // ParseURL parses an URL into options that can be used to connect to PostgreSQL. func ParseURL(sURL string) (*Options, error) { parsedUrl, err := url.Parse(sURL) if err != nil { return nil, err } // scheme if parsedUrl.Scheme != "postgres" && parsedUrl.Scheme != "postgresql" { return nil, errors.New("pg: invalid scheme: " + parsedUrl.Scheme) } // host and port options := &Options{ Addr: parsedUrl.Host, } if !strings.Contains(options.Addr, ":") { options.Addr = options.Addr + ":5432" } // username and password if parsedUrl.User != nil { options.User = parsedUrl.User.Username() if password, ok := parsedUrl.User.Password(); ok { options.Password = password } } if options.User == "" { options.User = "postgres" } // database if len(strings.Trim(parsedUrl.Path, "/")) > 0 { options.Database = parsedUrl.Path[1:] } else { return nil, errors.New("pg: database name not provided") } // ssl mode query, err := url.ParseQuery(parsedUrl.RawQuery) if err != nil { return nil, err } if sslMode, ok := query["sslmode"]; ok && len(sslMode) > 0 { switch sslMode[0] { case "allow", "prefer", "require": options.TLSConfig = &tls.Config{InsecureSkipVerify: true} case "disable": options.TLSConfig = nil default: return nil, errors.New(fmt.Sprintf("pg: sslmode '%v' is not supported", sslMode[0])) } } else { options.TLSConfig = &tls.Config{InsecureSkipVerify: true} } delete(query, "sslmode") if appName, ok := query["application_name"]; ok && len(appName) > 0 { options.ApplicationName = appName[0] } delete(query, "application_name") if len(query) > 0 { return nil, errors.New("pg: options other than 'sslmode' and 'application_name' are not supported") } return options, nil } func (opt *Options) getDialer() func() (net.Conn, error) { if opt.Dialer != nil { return func() (net.Conn, error) { return opt.Dialer(opt.Network, opt.Addr) } } return func() (net.Conn, error) { netDialer := &net.Dialer{ Timeout: opt.DialTimeout, KeepAlive: 5 * time.Minute, } return netDialer.Dial(opt.Network, opt.Addr) } } func newConnPool(opt *Options) *pool.ConnPool { return pool.NewConnPool(&pool.Options{ Dialer: opt.getDialer(), OnClose: func(cn *pool.Conn) error { return terminateConn(cn) }, PoolSize: opt.PoolSize, MinIdleConns: opt.MinIdleConns, MaxConnAge: opt.MaxConnAge, PoolTimeout: opt.PoolTimeout, IdleTimeout: opt.IdleTimeout, IdleCheckFrequency: opt.IdleCheckFrequency, }) }