85 lines
1.9 KiB
Go
85 lines
1.9 KiB
Go
package cluster
|
|
|
|
import (
|
|
"database/sql"
|
|
"fmt"
|
|
"strings"
|
|
|
|
_ "github.com/lib/pq"
|
|
|
|
"github.bus.zalan.do/acid/postgres-operator/pkg/spec"
|
|
"github.bus.zalan.do/acid/postgres-operator/pkg/util"
|
|
)
|
|
|
|
var createUserSQL = `SET LOCAL synchronous_commit = 'local'; CREATE ROLE "%s" %s %s;`
|
|
|
|
func (c *Cluster) pgConnectionString() string {
|
|
hostname := fmt.Sprintf("%s.%s.svc.cluster.local", c.Metadata.Name, c.Metadata.Namespace)
|
|
password := c.pgUsers[c.OpConfig.SuperUsername].Password
|
|
|
|
return fmt.Sprintf("host='%s' dbname=postgres sslmode=require user='%s' password='%s'",
|
|
hostname,
|
|
c.OpConfig.SuperUsername,
|
|
strings.Replace(password, "$", "\\$", -1))
|
|
}
|
|
|
|
func (c *Cluster) initDbConn() error {
|
|
//TODO: concurrent safe?
|
|
if c.pgDb == nil {
|
|
if c.pgDb == nil {
|
|
conn, err := sql.Open("postgres", c.pgConnectionString())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = conn.Ping()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
c.pgDb = conn
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (c *Cluster) createPgUser(user spec.PgUser) (isHuman bool, err error) {
|
|
var flags []string = user.Flags
|
|
|
|
if user.Password == "" {
|
|
isHuman = true
|
|
flags = append(flags, "SUPERUSER")
|
|
flags = append(flags, fmt.Sprintf("IN ROLE \"%s\"", c.OpConfig.PamRoleName))
|
|
} else {
|
|
isHuman = false
|
|
}
|
|
|
|
addLoginFlag := true
|
|
for _, v := range flags {
|
|
if v == "NOLOGIN" {
|
|
addLoginFlag = false
|
|
break
|
|
}
|
|
}
|
|
if addLoginFlag {
|
|
flags = append(flags, "LOGIN")
|
|
}
|
|
if !isHuman && user.MemberOf != "" {
|
|
flags = append(flags, fmt.Sprintf("IN ROLE \"%s\"", user.MemberOf))
|
|
}
|
|
userFlags := strings.Join(flags, " ")
|
|
userPassword := fmt.Sprintf("ENCRYPTED PASSWORD '%s'", util.PGUserPassword(user))
|
|
if user.Password == "" {
|
|
userPassword = "PASSWORD NULL"
|
|
}
|
|
query := fmt.Sprintf(createUserSQL, user.Name, userFlags, userPassword)
|
|
|
|
_, err = c.pgDb.Query(query) // TODO: Try several times
|
|
if err != nil {
|
|
err = fmt.Errorf("DB error: %s", err)
|
|
return
|
|
}
|
|
|
|
return
|
|
}
|