mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-04-20 05:59:22 +00:00
Resolves #7207 Add new configuration to make XORM work with a main and replicas database instances. The follow configuration parameters were added: - `HOST_PRIMARY` - `HOST_REPLICAS` - `LOAD_BALANCE_POLICY`. Options: - `"WeightRandom"` -> `xorm.WeightRandomPolicy` - `"WeightRoundRobin` -> `WeightRoundRobinPolicy` - `"LeastCon"` -> `LeastConnPolicy` - `"RoundRobin"` -> `xorm.RoundRobinPolicy()` - default: `xorm.RandomPolicy()` - `LOAD_BALANCE_WEIGHTS` Co-authored-by: pat-s <patrick.schratz@gmail.com@> Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/7212 Reviewed-by: Gusted <gusted@noreply.codeberg.org> Co-authored-by: pat-s <patrick.schratz@gmail.com> Co-committed-by: pat-s <patrick.schratz@gmail.com>
211 lines
5.5 KiB
Go
211 lines
5.5 KiB
Go
// Copyright 2019 The Gitea Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package setting
|
|
|
|
import (
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func Test_parsePostgreSQLHostPort(t *testing.T) {
|
|
tests := map[string]struct {
|
|
HostPort string
|
|
Host string
|
|
Port string
|
|
}{
|
|
"host-port": {
|
|
HostPort: "127.0.0.1:1234",
|
|
Host: "127.0.0.1",
|
|
Port: "1234",
|
|
},
|
|
"no-port": {
|
|
HostPort: "127.0.0.1",
|
|
Host: "127.0.0.1",
|
|
Port: "5432",
|
|
},
|
|
"ipv6-port": {
|
|
HostPort: "[::1]:1234",
|
|
Host: "::1",
|
|
Port: "1234",
|
|
},
|
|
"ipv6-no-port": {
|
|
HostPort: "[::1]",
|
|
Host: "::1",
|
|
Port: "5432",
|
|
},
|
|
"unix-socket": {
|
|
HostPort: "/tmp/pg.sock:1234",
|
|
Host: "/tmp/pg.sock",
|
|
Port: "1234",
|
|
},
|
|
"unix-socket-no-port": {
|
|
HostPort: "/tmp/pg.sock",
|
|
Host: "/tmp/pg.sock",
|
|
Port: "5432",
|
|
},
|
|
}
|
|
for k, test := range tests {
|
|
t.Run(k, func(t *testing.T) {
|
|
t.Log(test.HostPort)
|
|
host, port := parsePostgreSQLHostPort(test.HostPort)
|
|
assert.Equal(t, test.Host, host)
|
|
assert.Equal(t, test.Port, port)
|
|
})
|
|
}
|
|
}
|
|
|
|
func Test_getPostgreSQLConnectionString(t *testing.T) {
|
|
tests := []struct {
|
|
Host string
|
|
User string
|
|
Passwd string
|
|
Name string
|
|
SSLMode string
|
|
Output string
|
|
}{
|
|
{
|
|
Host: "", // empty means default
|
|
Output: "postgres://:@127.0.0.1:5432?sslmode=",
|
|
},
|
|
{
|
|
Host: "/tmp/pg.sock",
|
|
User: "testuser",
|
|
Passwd: "space space !#$%^^%^```-=?=",
|
|
Name: "gitea",
|
|
SSLMode: "false",
|
|
Output: "postgres://testuser:space%20space%20%21%23$%25%5E%5E%25%5E%60%60%60-=%3F=@:5432/gitea?host=%2Ftmp%2Fpg.sock&sslmode=false",
|
|
},
|
|
{
|
|
Host: "/tmp/pg.sock:6432",
|
|
User: "testuser",
|
|
Passwd: "pass",
|
|
Name: "gitea",
|
|
SSLMode: "false",
|
|
Output: "postgres://testuser:pass@:6432/gitea?host=%2Ftmp%2Fpg.sock&sslmode=false",
|
|
},
|
|
{
|
|
Host: "localhost",
|
|
User: "pgsqlusername",
|
|
Passwd: "I love Gitea!",
|
|
Name: "gitea",
|
|
SSLMode: "true",
|
|
Output: "postgres://pgsqlusername:I%20love%20Gitea%21@localhost:5432/gitea?sslmode=true",
|
|
},
|
|
{
|
|
Host: "localhost:1234",
|
|
User: "user",
|
|
Passwd: "pass",
|
|
Name: "gitea?param=1",
|
|
Output: "postgres://user:pass@localhost:1234/gitea?param=1&sslmode=",
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
connStr := getPostgreSQLConnectionString(test.Host, test.User, test.Passwd, test.Name, test.SSLMode)
|
|
assert.Equal(t, test.Output, connStr)
|
|
}
|
|
}
|
|
|
|
func getPostgreSQLEngineGroupConnectionStrings(primaryHost, replicaHosts, user, passwd, name, sslmode string) (string, []string) {
|
|
// Determine the primary connection string.
|
|
primary := primaryHost
|
|
if strings.TrimSpace(primary) == "" {
|
|
primary = "127.0.0.1:5432"
|
|
}
|
|
primaryConn := getPostgreSQLConnectionString(primary, user, passwd, name, sslmode)
|
|
|
|
// Build the replica connection strings.
|
|
replicaConns := []string{}
|
|
if strings.TrimSpace(replicaHosts) != "" {
|
|
// Split comma-separated replica host values.
|
|
hosts := strings.Split(replicaHosts, ",")
|
|
for _, h := range hosts {
|
|
trimmed := strings.TrimSpace(h)
|
|
if trimmed != "" {
|
|
replicaConns = append(replicaConns,
|
|
getPostgreSQLConnectionString(trimmed, user, passwd, name, sslmode))
|
|
}
|
|
}
|
|
}
|
|
|
|
return primaryConn, replicaConns
|
|
}
|
|
|
|
func Test_getPostgreSQLEngineGroupConnectionStrings(t *testing.T) {
|
|
tests := []struct {
|
|
primaryHost string // primary host setting (e.g. "localhost" or "[::1]:1234")
|
|
replicaHosts string // comma-separated replica hosts (e.g. "replica1,replica2:2345")
|
|
user string
|
|
passwd string
|
|
name string
|
|
sslmode string
|
|
outputPrimary string
|
|
outputReplicas []string
|
|
}{
|
|
{
|
|
// No primary override (empty => default) and no replicas.
|
|
primaryHost: "",
|
|
replicaHosts: "",
|
|
user: "",
|
|
passwd: "",
|
|
name: "",
|
|
sslmode: "",
|
|
outputPrimary: "postgres://:@127.0.0.1:5432?sslmode=",
|
|
outputReplicas: []string{},
|
|
},
|
|
{
|
|
// Primary set and one replica.
|
|
primaryHost: "localhost",
|
|
replicaHosts: "replicahost",
|
|
user: "user",
|
|
passwd: "pass",
|
|
name: "gitea",
|
|
sslmode: "disable",
|
|
outputPrimary: "postgres://user:pass@localhost:5432/gitea?sslmode=disable",
|
|
outputReplicas: []string{"postgres://user:pass@replicahost:5432/gitea?sslmode=disable"},
|
|
},
|
|
{
|
|
// Primary with explicit port; multiple replicas (one without and one with an explicit port).
|
|
primaryHost: "localhost:5433",
|
|
replicaHosts: "replica1,replica2:5434",
|
|
user: "test",
|
|
passwd: "secret",
|
|
name: "db",
|
|
sslmode: "require",
|
|
outputPrimary: "postgres://test:secret@localhost:5433/db?sslmode=require",
|
|
outputReplicas: []string{
|
|
"postgres://test:secret@replica1:5432/db?sslmode=require",
|
|
"postgres://test:secret@replica2:5434/db?sslmode=require",
|
|
},
|
|
},
|
|
{
|
|
// IPv6 addresses for primary and replica.
|
|
primaryHost: "[::1]:1234",
|
|
replicaHosts: "[::2]:2345",
|
|
user: "ipv6",
|
|
passwd: "ipv6pass",
|
|
name: "ipv6db",
|
|
sslmode: "disable",
|
|
outputPrimary: "postgres://ipv6:ipv6pass@[::1]:1234/ipv6db?sslmode=disable",
|
|
outputReplicas: []string{
|
|
"postgres://ipv6:ipv6pass@[::2]:2345/ipv6db?sslmode=disable",
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
primary, replicas := getPostgreSQLEngineGroupConnectionStrings(
|
|
test.primaryHost,
|
|
test.replicaHosts,
|
|
test.user,
|
|
test.passwd,
|
|
test.name,
|
|
test.sslmode,
|
|
)
|
|
assert.Equal(t, test.outputPrimary, primary)
|
|
assert.Equal(t, test.outputReplicas, replicas)
|
|
}
|
|
}
|