2023-12-28 08:29:18 +00:00
|
|
|
package postgresql
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"database/sql"
|
2024-05-30 09:39:15 +00:00
|
|
|
"errors"
|
2023-12-28 08:29:18 +00:00
|
|
|
"fmt"
|
2024-12-06 02:19:41 +00:00
|
|
|
"net/url"
|
2024-01-02 09:08:13 +00:00
|
|
|
"time"
|
|
|
|
|
2023-12-28 08:29:18 +00:00
|
|
|
"github.com/1Panel-dev/1Panel/backend/buserr"
|
|
|
|
"github.com/1Panel-dev/1Panel/backend/constant"
|
|
|
|
"github.com/1Panel-dev/1Panel/backend/utils/postgresql/client"
|
|
|
|
_ "github.com/jackc/pgx/v5/stdlib"
|
|
|
|
)
|
|
|
|
|
|
|
|
type PostgresqlClient interface {
|
|
|
|
Create(info client.CreateInfo) error
|
2024-01-08 09:40:42 +00:00
|
|
|
CreateUser(info client.CreateInfo, withDeleteDB bool) error
|
2023-12-28 08:29:18 +00:00
|
|
|
Delete(info client.DeleteInfo) error
|
2024-01-09 04:43:33 +00:00
|
|
|
ChangePrivileges(info client.Privileges) error
|
2023-12-28 08:29:18 +00:00
|
|
|
ChangePassword(info client.PasswordChangeInfo) error
|
|
|
|
|
|
|
|
Backup(info client.BackupInfo) error
|
|
|
|
Recover(info client.RecoverInfo) error
|
2024-01-02 09:08:13 +00:00
|
|
|
SyncDB() ([]client.SyncDBInfo, error)
|
2023-12-28 08:29:18 +00:00
|
|
|
Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewPostgresqlClient(conn client.DBInfo) (PostgresqlClient, error) {
|
|
|
|
if conn.From == "local" {
|
2024-01-07 14:52:41 +00:00
|
|
|
connArgs := []string{"exec", conn.Address, "psql", "-t", "-U", conn.Username, "-c"}
|
2024-01-02 09:08:13 +00:00
|
|
|
return client.NewLocal(connArgs, conn.Address, conn.Username, conn.Password, conn.Database), nil
|
2023-12-28 08:29:18 +00:00
|
|
|
}
|
2024-01-02 09:08:13 +00:00
|
|
|
|
2024-12-05 08:00:52 +00:00
|
|
|
// Escape username and password to handle special characters
|
2024-12-06 02:19:41 +00:00
|
|
|
escapedUsername := url.QueryEscape(conn.Username)
|
|
|
|
escapedPassword := url.QueryEscape(conn.Password)
|
2024-12-05 08:00:52 +00:00
|
|
|
|
|
|
|
connArgs := fmt.Sprintf("postgres://%s:%s@%s:%d/?sslmode=disable", escapedUsername, escapedPassword, conn.Address, conn.Port)
|
2023-12-28 08:29:18 +00:00
|
|
|
db, err := sql.Open("pgx", connArgs)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Duration(conn.Timeout)*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
if err := db.PingContext(ctx); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2024-05-30 09:39:15 +00:00
|
|
|
if errors.Is(ctx.Err(), context.DeadlineExceeded) {
|
2023-12-28 08:29:18 +00:00
|
|
|
return nil, buserr.New(constant.ErrExecTimeOut)
|
|
|
|
}
|
|
|
|
|
|
|
|
return client.NewRemote(client.Remote{
|
|
|
|
Client: db,
|
2024-01-08 10:21:33 +00:00
|
|
|
From: "remote",
|
2023-12-28 08:29:18 +00:00
|
|
|
Database: conn.Database,
|
|
|
|
User: conn.Username,
|
|
|
|
Password: conn.Password,
|
|
|
|
Address: conn.Address,
|
|
|
|
Port: conn.Port,
|
|
|
|
}), nil
|
|
|
|
}
|