2018-06-05 23:48:16 +03:00
|
|
|
package client
|
|
|
|
|
|
|
|
import (
|
2022-12-13 00:09:12 +03:00
|
|
|
"fmt"
|
2019-02-21 03:20:27 +03:00
|
|
|
"regexp"
|
2018-06-05 23:48:16 +03:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2019-02-25 20:43:04 +03:00
|
|
|
var (
|
|
|
|
// List of keywords that are not allowed in read-only mode
|
2023-11-03 05:17:46 +03:00
|
|
|
reRestrictedKeywords = regexp.MustCompile(`(?mi)\s?(CREATE|INSERT|UPDATE|DROP|DELETE|TRUNCATE|GRANT|OPEN|IMPORT|COPY)\s`)
|
2019-02-25 20:43:04 +03:00
|
|
|
|
|
|
|
// Comment regular expressions
|
|
|
|
reSlashComment = regexp.MustCompile(`(?m)/\*.+\*/`)
|
|
|
|
reDashComment = regexp.MustCompile(`(?m)--.+`)
|
2022-12-08 22:33:38 +03:00
|
|
|
|
|
|
|
// Postgres version signature
|
2022-12-13 00:09:12 +03:00
|
|
|
postgresSignature = regexp.MustCompile(`(?i)postgresql ([\d\.]+)\s?`)
|
|
|
|
postgresDumpSignature = regexp.MustCompile(`\s([\d\.]+)\s?`)
|
|
|
|
postgresType = "PostgreSQL"
|
2022-12-08 22:33:38 +03:00
|
|
|
|
|
|
|
// Cockroach version signature
|
|
|
|
cockroachSignature = regexp.MustCompile(`(?i)cockroachdb ccl v([\d\.]+)\s?`)
|
|
|
|
cockroachType = "CockroachDB"
|
2019-02-25 20:43:04 +03:00
|
|
|
)
|
2019-02-21 03:20:27 +03:00
|
|
|
|
2022-12-13 00:09:12 +03:00
|
|
|
// Get major and minor version components
|
2018-06-05 23:48:16 +03:00
|
|
|
// Example: 10.2.3.1 -> 10.2
|
2022-12-13 00:09:12 +03:00
|
|
|
func getMajorMinorVersion(str string) (major int, minor int) {
|
2018-06-05 23:48:16 +03:00
|
|
|
chunks := strings.Split(str, ".")
|
2022-12-13 00:09:12 +03:00
|
|
|
fmt.Sscanf(chunks[0], "%d", &major)
|
|
|
|
if len(chunks) > 1 {
|
|
|
|
fmt.Sscanf(chunks[1], "%d", &minor)
|
2018-06-05 23:48:16 +03:00
|
|
|
}
|
2022-12-13 00:09:12 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get short version from the string
|
|
|
|
// Example: 10.2.3.1 -> 10.2
|
|
|
|
func getMajorMinorVersionString(str string) string {
|
|
|
|
major, minor := getMajorMinorVersion(str)
|
|
|
|
return fmt.Sprintf("%d.%d", major, minor)
|
2018-06-05 23:48:16 +03:00
|
|
|
}
|
2019-02-21 03:20:27 +03:00
|
|
|
|
2022-12-08 22:33:38 +03:00
|
|
|
func detectServerTypeAndVersion(version string) (bool, string, string) {
|
|
|
|
version = strings.TrimSpace(version)
|
|
|
|
|
|
|
|
// Detect postgresql
|
|
|
|
matches := postgresSignature.FindAllStringSubmatch(version, 1)
|
|
|
|
if len(matches) > 0 {
|
|
|
|
return true, postgresType, matches[0][1]
|
|
|
|
}
|
|
|
|
|
|
|
|
// Detect cockroachdb
|
|
|
|
matches = cockroachSignature.FindAllStringSubmatch(version, 1)
|
|
|
|
if len(matches) > 0 {
|
|
|
|
return true, cockroachType, matches[0][1]
|
|
|
|
}
|
|
|
|
|
|
|
|
return false, "", ""
|
|
|
|
}
|
|
|
|
|
2022-12-13 00:09:12 +03:00
|
|
|
// detectDumpVersion parses out version from `pg_dump -V` command.
|
|
|
|
func detectDumpVersion(version string) (bool, string) {
|
|
|
|
matches := postgresDumpSignature.FindAllStringSubmatch(version, 1)
|
|
|
|
if len(matches) > 0 {
|
|
|
|
return true, matches[0][1]
|
|
|
|
}
|
|
|
|
return false, ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func checkVersionRequirement(client, server string) bool {
|
|
|
|
clientMajor, clientMinor := getMajorMinorVersion(client)
|
|
|
|
serverMajor, serverMinor := getMajorMinorVersion(server)
|
|
|
|
|
|
|
|
if serverMajor < 10 {
|
|
|
|
return clientMajor >= serverMajor && clientMinor >= serverMinor
|
|
|
|
}
|
|
|
|
|
|
|
|
return clientMajor >= serverMajor
|
|
|
|
}
|
|
|
|
|
2019-02-21 03:20:27 +03:00
|
|
|
// containsRestrictedKeywords returns true if given keyword is not allowed in read-only mode
|
|
|
|
func containsRestrictedKeywords(str string) bool {
|
2019-02-25 20:43:04 +03:00
|
|
|
str = reSlashComment.ReplaceAllString(str, "")
|
|
|
|
str = reDashComment.ReplaceAllString(str, "")
|
|
|
|
|
|
|
|
return reRestrictedKeywords.MatchString(str)
|
2019-02-21 03:20:27 +03:00
|
|
|
}
|
2021-12-29 20:03:50 +03:00
|
|
|
|
|
|
|
func hasBinary(data string, checkLen int) bool {
|
|
|
|
for idx, chr := range data {
|
|
|
|
if int(chr) < 32 || int(chr) > 126 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if idx >= checkLen {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|