package io.vertx.pgclient.impl;
import io.vertx.pgclient.SslMode;
import io.vertx.core.json.JsonObject;
import java.io.UnsupportedEncodingException;
import java.net.URLDecoder;
import java.util.HashMap;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import static java.lang.Integer.parseInt;
import static java.lang.String.format;
public class PgConnectionUriParser {
private static final String SCHEME_DESIGNATOR_REGEX = "postgre(s|sql)://";
private static final String USER_INFO_REGEX = "((?<userinfo>[a-zA-Z0-9\\-._~%!]+(:[a-zA-Z0-9\\-._~%!]+)?)@)?";
private static final String NET_LOCATION_REGEX = "(?<netloc>[0-9.]+|\\[[a-zA-Z0-9:]+]|[a-zA-Z0-9\\-._~%]+)?";
private static final String PORT_REGEX = "(:(?<port>\\d+))?";
private static final String DATABASE_REGEX = "(/(?<database>[a-zA-Z0-9\\-._~%!]+))?";
private static final String PARAMS_REGEX = "(\\?(?<params>.*))?";
private static final String FULL_URI_REGEX = "^"
+ SCHEME_DESIGNATOR_REGEX
+ USER_INFO_REGEX
+ NET_LOCATION_REGEX
+ PORT_REGEX
+ DATABASE_REGEX
+ PARAMS_REGEX
+ "$";
public static JsonObject parse(String connectionUri) {
try {
JsonObject configuration = new JsonObject();
doParse(connectionUri, configuration);
return configuration;
} catch (Exception e) {
throw new IllegalArgumentException("Cannot parse invalid connection URI: " + connectionUri, e);
}
}
private static void doParse(String connectionUri, JsonObject configuration) {
Pattern pattern = Pattern.compile(FULL_URI_REGEX);
Matcher matcher = pattern.matcher(connectionUri);
if (matcher.matches()) {
parseUserAndPassword(matcher.group("userinfo"), configuration);
parseNetLocation(matcher.group("netloc"), configuration);
parsePort(matcher.group("port"), configuration);
parseDatabaseName(matcher.group("database"), configuration);
parseParameters(matcher.group("params"), configuration);
} else {
throw new IllegalArgumentException("Wrong syntax of connection URI");
}
}
private static void parseUserAndPassword(String userInfo, JsonObject configuration) {
if (userInfo == null || userInfo.isEmpty()) {
return;
}
if (occurExactlyOnce(userInfo, ":")) {
int index = userInfo.indexOf(":");
String user = userInfo.substring(0, index);
if (user.isEmpty()) {
throw new IllegalArgumentException("Can not only specify the password without a concrete user");
}
String password = userInfo.substring(index + 1);
configuration.put("user", decodeUrl(user));
configuration.put("password", decodeUrl(password));
} else if (!userInfo.contains(":")) {
configuration.put("user", decodeUrl(userInfo));
} else {
throw new IllegalArgumentException("Can not use multiple delimiters to delimit user and password");
}
}
private static void parseNetLocation(String hostInfo, JsonObject configuration) {
if (hostInfo == null || hostInfo.isEmpty()) {
return;
}
parseNetLocationValue(decodeUrl(hostInfo), configuration);
}
private static void parsePort(String portInfo, JsonObject configuration) {
if (portInfo == null || portInfo.isEmpty()) {
return;
}
int port;
try {
port = parseInt(decodeUrl(portInfo));
} catch (NumberFormatException e) {
throw new IllegalArgumentException("The port must be a valid integer");
}
if (port > 65535 || port <= 0) {
throw new IllegalArgumentException("The port can only range in 1-65535");
}
configuration.put("port", port);
}
private static void parseDatabaseName(String databaseInfo, JsonObject configuration) {
if (databaseInfo == null || databaseInfo.isEmpty()) {
return;
}
configuration.put("database", decodeUrl(databaseInfo));
}
private static void parseParameters(String parametersInfo, JsonObject configuration) {
if (parametersInfo == null || parametersInfo.isEmpty()) {
return;
}
Map<String, String> properties = new HashMap<>();
for (String parameterPair : parametersInfo.split("&")) {
if (parameterPair.isEmpty()) {
continue;
}
int indexOfDelimiter = parameterPair.indexOf("=");
if (indexOfDelimiter < 0) {
throw new IllegalArgumentException(format("Missing delimiter '=' of parameters \"%s\" in the part \"%s\"", parametersInfo, parameterPair));
} else {
String key = parameterPair.substring(0, indexOfDelimiter).toLowerCase();
String value = decodeUrl(parameterPair.substring(indexOfDelimiter + 1).trim());
switch (key) {
case "port":
parsePort(value, configuration);
break;
case "host":
parseNetLocationValue(value, configuration);
break;
case "hostaddr":
configuration.put("host", value);
break;
case "user":
configuration.put("user", value);
break;
case "password":
configuration.put("password", value);
break;
case "dbname":
configuration.put("database", value);
break;
case "sslmode":
configuration.put("sslMode", SslMode.of(value));
break;
case "application_name":
properties.put("application_name", value);
break;
case "fallback_application_name":
properties.put("fallback_application_name", value);
break;
case "search_path":
properties.put("search_path", value);
break;
default:
configuration.put(key, value);
break;
}
}
}
if (!properties.isEmpty()) {
configuration.put("properties", properties);
}
}
private static void parseNetLocationValue(String hostValue, JsonObject configuration) {
if (isRegardedAsIpv6Address(hostValue)) {
configuration.put("host", hostValue.substring(1, hostValue.length() - 1));
} else {
configuration.put("host", hostValue);
}
}
private static boolean isRegardedAsIpv6Address(String hostAddress) {
return hostAddress.startsWith("[") && hostAddress.endsWith("]");
}
private static String decodeUrl(String url) {
try {
return URLDecoder.decode(url, "UTF-8");
} catch (UnsupportedEncodingException e) {
throw new IllegalArgumentException("The connection uri contains unknown characters that can not be resolved.");
}
}
private static boolean occurExactlyOnce(String uri, String character) {
return uri.contains(character) && uri.indexOf(character) == uri.lastIndexOf(character);
}
}