org.swisspush.gateleen.routing.RuleFactory Maven / Gradle / Ivy
package org.swisspush.gateleen.routing;
import io.vertx.core.buffer.Buffer;
import io.vertx.core.http.HttpClientOptions;
import io.vertx.core.json.JsonArray;
import io.vertx.core.json.JsonObject;
import io.vertx.core.net.ProxyOptions;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.swisspush.gateleen.core.http.HeaderFunction;
import org.swisspush.gateleen.core.http.HeaderFunctions;
import org.swisspush.gateleen.core.util.StringUtils;
import org.swisspush.gateleen.core.validation.ValidationResult;
import org.swisspush.gateleen.validation.RegexpValidator;
import org.swisspush.gateleen.validation.ValidationException;
import org.swisspush.gateleen.validation.Validator;
import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* RuleFactory is used to create rule objects from their text representation.
*/
public class RuleFactory {
private Logger log = LoggerFactory.getLogger(RuleFactory.class);
private Pattern urlParsePattern = Pattern.compile("^(?https?)://(?[^/:]+)(:(?\\$?[0-9]+))?(?/.*)$");
private final Map properties;
private String routingRulesSchema;
public RuleFactory(Map properties, String routingRulesSchema) {
this.properties = properties;
this.routingRulesSchema = routingRulesSchema;
}
public List parseRules(Buffer buffer, int routeMultiplier) throws ValidationException {
String replacedConfig;
try {
replacedConfig = StringUtils.replaceWildcardConfigs(buffer.toString("UTF-8"), properties);
} catch (Exception e) {
throw new ValidationException(e);
}
ValidationResult validationResult = Validator.validateStatic(Buffer.buffer(replacedConfig), routingRulesSchema, log);
if (validationResult.isSuccess()) {
return createRules(new JsonObject(replacedConfig), routeMultiplier);
} else {
throw new ValidationException(validationResult);
}
}
public List createRules(JsonObject rules, int routeMultiplier) throws ValidationException {
Set metricNames = new HashSet<>();
List result = new ArrayList<>();
for (String urlPattern : rules.fieldNames()) {
log.debug("Creating a new rule-object for URL pattern {}", urlPattern);
Rule ruleObj = new Rule();
ruleObj.setUrlPattern(urlPattern);
JsonObject rule = rules.getJsonObject(urlPattern);
String headersFilter = rule.getString("headersFilter");
if(headersFilter != null) {
Pattern headersFilterPattern = RegexpValidator.throwIfPatternInvalid(headersFilter);
ruleObj.setHeadersFilterPattern(headersFilterPattern);
}
String targetUrl = rule.getString("url");
String path = rule.getString("path");
prepareUrl(urlPattern, ruleObj, targetUrl, path);
String metricName = rule.getString("metricName");
if (metricName != null) {
if(metricNames.contains(metricName)){
throw new ValidationException("Property 'metricName' must be unique. There are multiple rules with metricName '" + metricName + "'");
} else {
metricNames.add(metricName);
ruleObj.setMetricName(metricName);
}
}
ruleObj.setTimeout(1000 * rule.getInteger(Rule.CONNECTION_TIMEOUT_SEC_PROPERTY_NAME, Rule.CONNECTION_TIMEOUT_SEC_DEFAULT_VALUE));
ruleObj.setKeepAliveTimeout(rule.getInteger("keepAliveTimeout", HttpClientOptions.DEFAULT_KEEP_ALIVE_TIMEOUT));
ruleObj.setPoolSize(rule.getInteger(Rule.CONNECTION_POOL_SIZE_PROPERTY_NAME, Rule.CONNECTION_POOL_SIZE_DEFAULT_VALUE));
int originalPoolSize = ruleObj.getPoolSize();
int appliedPoolSize = evaluatePoolSize(originalPoolSize, routeMultiplier);
ruleObj.setPoolSize(appliedPoolSize);
log.debug("Original pool size is {}, applied size is {}", originalPoolSize, appliedPoolSize);
ruleObj.setMaxWaitQueueSize(rule.getInteger(Rule.MAX_WAIT_QUEUE_SIZE_PROPERTY_NAME, Rule.MAX_WAIT_QUEUE_SIZE_DEFAULT_VALUE));
ruleObj.setKeepAlive(rule.getBoolean("keepAlive", true));
ruleObj.setExpandOnBackend(rule.getBoolean("expandOnBackend", false));
ruleObj.setDeltaOnBackend(rule.getBoolean("deltaOnBackend", false));
ruleObj.setStorageExpand(rule.getBoolean("storageExpand", false));
ruleObj.setLogExpiry(rule.getInteger("logExpiry", 4 * 3600));
JsonArray methods = rule.getJsonArray("methods");
if (methods != null) {
String[] methodsArray = new String[methods.size()];
for (int i=0; i < methods.size(); i++) {
methodsArray[i] = methods.getString(i);
}
ruleObj.setMethods(methodsArray);
}
JsonArray profile = rule.getJsonArray("profile");
if (profile != null) {
String[] profileArray = new String[profile.size()];
for (int i=0; i < profile.size(); i++) {
profileArray[i] = profile.getString(i);
}
ruleObj.setProfile(profileArray);
log.debug("The profile-array is set. Those profile-information will be sent: {}", Arrays.toString(ruleObj.getProfile()));
} else {
log.debug("The profile-array is not set. So won't send any profile-information when using this rule.");
}
setStorage(ruleObj, rule, path);
setTranslateStatus(ruleObj, rule);
setStaticHeaders(ruleObj, rule);
setProxyOptions(ruleObj, rule);
setAuthentication(ruleObj, rule);
result.add(ruleObj);
}
return result;
}
public static int evaluatePoolSize(int originalPoolSize, int routeMultiplier) {
return ceilDiv(originalPoolSize, routeMultiplier);
}
private static int ceilDiv(int x, int y){
return -Math.floorDiv(-x,y);
}
private void setStorage(Rule ruleObj, JsonObject rule, String path) throws ValidationException {
String ruleStorage = rule.getString("storage");
if (ruleStorage != null) {
if (path == null) {
throw new ValidationException("For storage routing, 'path' must be specified.");
}
ruleObj.setScheme("storage");
ruleObj.setStorage(ruleStorage);
}
}
private void setTranslateStatus(Rule ruleObj, JsonObject rule) {
JsonObject translateStatus = rule.getJsonObject("translateStatus");
if (translateStatus != null) {
ruleObj.addTranslateStatus(new LinkedHashMap<>());
for (String pattern : translateStatus.fieldNames()) {
ruleObj.getTranslateStatus().put(Pattern.compile(pattern), translateStatus.getInteger(pattern));
}
}
}
private void setProxyOptions(Rule ruleObj, JsonObject rule) {
JsonObject proxyOptions = rule.getJsonObject("proxyOptions");
if(proxyOptions != null){
ruleObj.setProxyOptions(new ProxyOptions(proxyOptions));
}
}
private void setAuthentication(Rule ruleObj, JsonObject rule) throws ValidationException {
JsonObject basicAuth = rule.getJsonObject("basicAuth");
String oAuthId = rule.getString("oAuthId");
if (basicAuth != null && oAuthId != null) {
throw new ValidationException("Either 'basicAuth' or 'oAuthId' can be given, not both");
}
if (basicAuth != null) {
ruleObj.setBasicAuthUsername(basicAuth.getString("username"));
ruleObj.setBasicAuthPassword(basicAuth.getString("password"));
} else if (oAuthId != null) {
ruleObj.setOAuthId(oAuthId);
}
}
@Deprecated
private void setStaticHeaders(Rule ruleObj, JsonObject rule) {
final JsonArray headers = rule.getJsonArray("headers");
if (headers != null) {
final HeaderFunction headerFunction = HeaderFunctions.parseFromJson(headers);
ruleObj.setHeaderFunction(headerFunction);
return;
}
// in previous Gateleen versions we only had the "staticHeaders" to unconditionally add headers with fix values
// We now have a more dynamic concept of a "manipulator chain" - which is also configured different in JSON syntax
// For backward compatibility we still parse the old "staticHeaders" - but now create a manipulator chain accordingly
JsonObject staticHeaders = rule.getJsonObject("staticHeaders");
if (staticHeaders != null) {
log.warn("you use the deprecated \"staticHeaders\" syntax in your routing rule JSON ({}). Please migrate to the more flexible \"headers\" syntax", rule);
ruleObj.setHeaderFunction(HeaderFunctions.parseStaticHeadersFromJson(staticHeaders));
}
}
private void prepareUrl(String urlPattern, Rule ruleObj, String targetUrl, String path) throws ValidationException {
if (targetUrl != null || path != null) {
if (targetUrl != null && path != null) {
throw new ValidationException("Either 'url' or 'path' must be given, not both");
}
if (targetUrl != null) {
Matcher urlMatcher = urlParsePattern.matcher(targetUrl);
if (!urlMatcher.matches()) {
throw new ValidationException("Invalid url for pattern " + urlPattern + ": " + targetUrl);
}
ruleObj.setHost(urlMatcher.group("host"));
ruleObj.setScheme(urlMatcher.group("scheme"));
ruleObj.setPort(ruleObj.getScheme().equals("https") ? 443 : 80);
String portString = urlMatcher.group("port");
if (portString != null) {
if(portString.startsWith("$")){
ruleObj.setPortWildcard(portString);
} else {
ruleObj.setPort(Integer.parseInt(portString));
}
}
ruleObj.setPath(urlMatcher.group("path"));
} else {
if (!path.startsWith("/")) {
throw new ValidationException("Illegal value for 'path', it must be a path starting with slash");
}
ruleObj.setPath(path);
ruleObj.setHost("localhost");
ruleObj.setScheme("local");
}
} else {
// preventing a nullpointer
ruleObj.setScheme("null");
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy